查询
本部分是对
「Rust入门系列」Rust 中使用 MySQL的学习与记录
- 经常使用的时间处理库:
chrono
- 流式查询使用:
query_iter
- 输出到Vec使用:
query
- 映射到结构体使用:
query_map
- 获取单条数据使用:
query_first
- 命名参数查询使用:
exec_first
1 2 3 4 5 6 7 8 9 10 11 12
| CREATE TABLE `student` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(128) NOT NULL, `age` int(11) NOT NULL, `id_card` varchar(128) NOT NULL, `last_update` date NOT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
insert into student (name, age, id_card, last_update) values ('张三', 23, '123456789X', CURRENT_DATE()); insert into student (name, age, id_card, last_update) values ('李四', 24, '8382353902', CURRENT_DATE())
|
注意,mysql这个crate新版本demo有问题,文档的更新速度跟不上代码的修改脚步..
需要指定版本:
1 2
| [dependencies] mysql = "20.0.0"
|
流式查询
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| use chrono::prelude::*; use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
conn.query_iter("select * from student") .unwrap() .for_each(|row| { let r: (i32, String, i32, String, NaiveDate) = from_row(row.unwrap()); println!("{}, {},{},{}, {:?}", r.0, r.1, r.2, r.3, r.4); }); }
|
row的类型是mysql_common::row::Row,其把数据以字节的形式存储。
所以需将低级的字节转换成想要的类型 如i32,String,这里使用了from_row。注意,转换后的数据以元组的形式返回,其中每一项和选择列的顺序相同。
输出:
1 2
| 1, 张三,23,123456789X, 2022-04-26 2, 李四,24,8382353902, 2022-04-26
|
聚合查询结果
其实还可以将查询结果收集到Vec中。 Vec中的每个元素都是一个元组。
query函数已经将字节转换为选择的数据类型,因此不需要再转换了。 需要注意的是,这里必须明确元组的数据类型(如此处是 Vec<(i32, String, i32, String, NaiveDate)>)。 否则,编译器没办法做转换。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| use chrono::prelude::*; use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
let res: Vec<(i32, String, i32, String, NaiveDate)> = conn.query("select * from student").unwrap(); for r in res { println!("{}, {},{},{}, {:?}", r.0, r.1, r.2, r.3, r.4); } }
|
映射结果到结构体
如果表的列数很多,使用元组容易混淆,更普遍的做法是定义一个结构体。
如下定义一个Student结构体, 然后可以用query_map将查询结果映射到Student中。
不需要指定数据类型,编译器会根据Student类型自动推导
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
| use chrono::prelude::*; use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
struct Student { id: u64, name: String, age: u16, id_card: String, last_changed_on: NaiveDate, }
let res = conn.query_map( "select * from student", |(id, name, age, id_card, update)| Student { id: id, name: name, age: age, id_card: id_card, last_changed_on: update, }, ).expect("Query failed.");
for i in res { println!( "{}, {},{},{}, {:?}", i.id, i.name, i.age, i.id_card, i.last_changed_on ) } }
|
单条数据查询
查询特定数据行,可能会出现下面几种情况:
所以,使用query_first函数返回的是Option的结果。 需要将其解包两次才可以获取实际的行数据:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
| use chrono::prelude::*; use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
struct Student { id: u64, name: String, age: u16, id_card: String, last_changed_on: NaiveDate, }
let res = conn.query_first("select * from student where name = '张三'") .map( |row| { row.map(|(id, name, age, id_card, update)| Student { id: id, name: name, age: age, id_card: id_card, last_changed_on: update, }) }, );
match res.unwrap() { Some(student) => println!( "{}, {},{},{}, {:?}", student.id, student.name, student.age, student.id_card, student.last_changed_on ), None => println!("Sorry no student found."), } }
|
命名参数的使用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49
| use chrono::prelude::*; use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
struct Student { id: u64, name: String, age: u16, id_card: String, last_changed_on: NaiveDate, }
let res = conn .exec_first( "select * from student where name = :name", params! { "name" => "李四" }, ) .map( |row| { row.map(|(id, name, age, id_card, update)| Student { id: id, name: name, age: age, id_card: id_card, last_changed_on: update, }) }, );
match res.unwrap() { Some(student) => println!( "{}, {},{},{}, {:?}", student.id, student.name, student.age, student.id_card, student.last_changed_on ), None => println!("Sorry no student found."), }
}
|
写操作
本部分是对
Rust使用MySQL数据库02的学习与记录
- 插入数据使用
conn.exec_drop()
- 使用预编译语句插入大量数据,
conn.prep()
- 使用
conn.last_insert_id()可以获取主键
- 更新和删除也使用
conn.prep和conn.exec_drop
插入新数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| use chrono::prelude::*;
use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
conn.exec_drop( "INSERT INTO student (name, age, id_card, last_update) VALUES (:name, :age, :id_card, :last_update)", params! { "name" => "王五", "age" => 28, "id_card" => "66666688", "last_update" => today(), }).unwrap(); }
fn today() -> NaiveDate { let l = Local::today();
NaiveDate::from_ymd(l.year(), l.month(), l.day()) }
|
和上面一节一样,命名参数在这里使用了params宏的语法
exec_drop方法中的drop表示没有返回结果
用于执行插入/更新/删除的sql
使用预编译语句
使用conn.prep将sql编译成预编译语句。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| use chrono::prelude::*;
use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
let stmt = conn.prep("INSERT INTO student (name, age, id_card, last_update) VALUES (:name, :age, :id_card, :last_update)") .unwrap();
for i in 1..10 { conn.exec_drop(&stmt, params! { "name" => "dashen", "age" => 18 + i, "id_card" => "1234565X", "last_update" => NaiveDate::from_ymd(2017, 05, 04), }).unwrap() } }
|
获取生成的主键id
可以通过conn.last_insert_id()方法获取到新记录的主键id,该方法将返回的一个类型为u64
的值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| use chrono::prelude::*;
use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
conn.exec_drop("INSERT INTO student (name, age, id_card, last_update) VALUES (:name, :age, :id_card, :last_update)", params! { "name" => "fliter", "age" => 29, "id_card" => "88888888", "last_update" => NaiveDate::from_ymd(2022, 05, 04), }).unwrap();
println!("新插入的记录的主键为: {}", conn.last_insert_id()) }
|
更新和删除
类似于插入操作
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| use chrono::prelude::*;
use mysql::*; use mysql::prelude::*;
fn main() { let url = "mysql://root:12345678@localhost:3306/shuang"; let pool = Pool::new(url).unwrap();
let mut conn = pool.get_conn().unwrap();
let stmt = conn.prep("update student set name=:name, last_update=:last_update where id=:id") .unwrap();
conn.exec_drop(&stmt, params! { "name" => "新名字", "last_update" => NaiveDate::from_ymd(2038, 12, 31), "id" => 10, }).unwrap();
let stmt = conn.prep("delete from student where id=:id").unwrap();
conn.exec_drop(&stmt, params! { "id" => 12, }).unwrap(); }
|
原文链接: https://dashen.tech/2017/05/04/Rust操作MySQL/
版权声明: 转载请注明出处.