MySQL和PostgreSQL对比与选型
本章基于个人经验,对两款数据库产品的某些重要特性的差异进行对比,并对选型提供参考意见。使用 Rust 的 u32 让 PostgreSQL 实现 MySQL 的 INT UNSIGNED
在上一章我们讨论到 PostgreSQL 没有 UNSIGNED 的问题,本章我们试图通过 rust 的 u32 来映射 PostgreSQL 的 int。MySQL实现BOOLEAN
茴香豆有多少种写法?这是上学时,鲁迅借着孔乙己的手,告诉了我们答案。本章我们将讨论在 MySQL 实现 bool 有多少种方法。【实战】部署 MySQL 8 主从复制和读写分离
利用主从复制,可以实现 MySQL 的读写分离、热备份等。本章将带你实战 MySQL 的一主二从。
使用 Rust 的 u32 让 PostgreSQL 实现 MySQL 的 INT UNSIGNED
- 51577
- 2022-10-12 19:02:57
在上一章我们讨论到 PostgreSQL 没有 UNSIGNED 的问题,本章我们试图通过 rust 的 u32 来映射 PostgreSQL 的 int。
i32
和 u32
的转换
在 Rust 和 PostgreSQL 中,Rust 的 i32
对应 PostgreSQL 的 INTEGER(INT)
,它们都是有符号的 32 位整数。而 Rust 的 u32
是无符号的 32 位整数。
无论 Rust 的 i32
还是 PostgreSQL 的 INT
,当它们的值达到取值范围的最大值时,它们会溢出为最小值。
#[test]
fn test_max_i32_plus_1() {
let max_i32 = i32::max_value();
let min_i32 = i32::min_value();
let from_max_i32 = (max_i32 as u32) + 1;
let from_min_i32 = min_i32 as u32;
assert!(from_max_i32 == from_min_i32);
}
使用 Rust 的 u32
让 PostgreSQL 实现 MySQL 的 INT UNSIGNED
利用上述的方法,我们可以实现 MySQL 的 INT UNSIGNED
。
代码演示
fn main() {
println!("Hello, world!");
}
pub struct PgIntAndRustU32 {
pub hit: u32,
}
pub struct PgSerialAndRustU32 {
pub id: u32,
}
#[cfg(test)]
mod test {
use super::*;
use sqlx::{postgres::PgPoolOptions, Row};
#[test]
fn test_max_i32_plus_1() {
let max_i32 = i32::max_value();
let min_i32 = i32::min_value();
let from_max_i32 = (max_i32 as u32) + 1;
let from_min_i32 = min_i32 as u32;
assert!(from_max_i32 == from_min_i32);
}
#[tokio::test]
async fn test_pg_int_and_rust_u32() {
let pool = PgPoolOptions::new().max_connections(3).connect("postgres://axum_rs:[email protected]/axum_rs").await.unwrap();
let max_i32 = i32::max_value();
let i: u32 = (max_i32 as u32) + 1;
sqlx::query("INSERT INTO test_int_to_u32(hit) VALUES ($1)")
.bind(i as i32)
.execute(&pool)
.await
.unwrap();
}
#[tokio::test]
async fn test_pg_int_and_rust_u32_read() {
let pool = PgPoolOptions::new().max_connections(3).connect("postgres://axum_rs:[email protected]/axum_rs").await.unwrap();
let row = sqlx::query("SELECT hit FROM test_int_to_u32")
.fetch_one(&pool)
.await
.unwrap();
let hit: i32 = row.get(0);
let obj = PgIntAndRustU32 { hit: hit as u32 };
assert!(obj.hit == (i32::max_value() as u32) + 1);
assert!(obj.hit as i32 == i32::min_value());
}
#[tokio::test]
async fn test_pg_serial_and_rust_u32() {
let pool = PgPoolOptions::new().max_connections(3).connect("postgres://axum_rs:[email protected]/axum_rs").await.unwrap();
let max_i32 = i32::max_value();
let i: u32 = (max_i32 as u32) + 1;
sqlx::query("INSERT INTO test_serial_to_u32(id) VALUES ($1)")
.bind(i as i32)
.execute(&pool)
.await
.unwrap();
}
#[tokio::test]
async fn test_pg_serial_and_rust_u32read() {
let pool = PgPoolOptions::new().max_connections(3).connect("postgres://axum_rs:[email protected]/axum_rs").await.unwrap();
let row = sqlx::query("SELECT hit FROM test_serial_to_u32")
.fetch_one(&pool)
.await
.unwrap();
let id: i32 = row.get(0);
let obj = PgSerialAndRustU32 { id: id as u32 };
assert!(obj.id == (i32::max_value() as u32) + 1);
assert!(obj.id as i32 == i32::min_value());
}
}