Rust语言从入门到精通系列 - Rust语言MySQL实战

1581 字
8 分钟
Rust语言从入门到精通系列 - Rust语言MySQL实战

MySQL是一个广泛使用的关系型数据库,Rust作为一门相对较新的系统级编程语言,具有C语言般的高性能、安全、并发等特性,因此与MySQL一起使用是一种非常有趣的选择。在本教程中,我们将手把手地展示如何在Rust中连接和使用MySQL数据库。

安装 mysql 模块#

这里我们假设你已经安装了Rust编程语言工具链,在本教程中,我们将使用mysql crate来连接和使用MySQL数据库。要安装mysql crate,我们可以使用Rust语言包管理器cargo,只需在终端中输入以下命令:

cargo install mysql

安装成功后,我们可以开始尝试连接MySQL数据库了。

连接MySQL数据库#

首先,我们需要安装和配置MySQL数据库,以便在Rust程序中进行连接。安装和配置MySQL在此处不做叙述。

在Rust程序中使用mysql crate库连接MySQL数据库,需要进行以下步骤:

  1. 导入mysql crate
  2. 使用mysql::OptsBuilder设置MySQL连接选项
  3. 使用mysql::Pool::new创建MySQL连接池
  4. 使用pool.get_conn()获取MySQL连接,并进行一些操作,例如插入、查询等
  5. 使用pool.disconnect()断开MySQL连接

下面是连接MySQL数据库的示例代码:

use mysql::*;
fn main() {
let opts = OptsBuilder::new()
.ip_or_hostname(Some("localhost"))
.user(Some("root"))
.pass(Some("password"))
.db_name(Some("test"))
.tcp_port(3306);
let pool = Pool::new(opts).unwrap();
let mut conn = pool.get_conn().unwrap();
let result = conn.query_first("SELECT * FROM users").unwrap();
for row in result {
let name: String = row.get("name").unwrap();
let age: i32 = row.get("age").unwrap();
println!("{} is {} years old", name, age);
}
pool.disconnect().unwrap();
}

以上代码创建了一个MySQL连接池,并从连接池中获取一个MySQL连接,查询了一个名为users的表,并将结果作为元素进行遍历。

Rust使用MySQL的进阶用法#

事务(Transaction)#

为了保证MySQL数据库中的数据一致性,我们通常需要使用事务(Transaction)。在Rust中,可以使用MySQL的事务功能并结合mysql::Transaction来实现。

使用以下代码示例可以体验事务的实现:

use mysql::*;
fn main() {
let opts = OptsBuilder::new()
.ip_or_hostname(Some("localhost"))
.user(Some("root"))
.pass(Some("password"))
.db_name(Some("test"))
.tcp_port(3306);
let pool = Pool::new(opts).unwrap();
let mut conn = pool.get_conn().unwrap();
// Start a transaction
let mut transaction = conn.start_transaction(TxOpts::default()).unwrap();
// Insert data into a table
transaction .prep_exec("INSERT INTO users (name, age) VALUES (?, ?)", ("Alice", 23)).unwrap();
transaction.prep_exec("INSERT INTO users (name, age) VALUES (?, ?)", ("Bob", 25)).unwrap();
// Commit a transaction
transaction.commit().unwrap();
// Select data from a table let result = conn.query("SELECT * FROM users").unwrap();
for row in result {
let name: String = row.get("name").unwrap();
let age: i32 = row.get("age").unwrap();
println!("{} is {} years old", name, age);
}
pool.disconnect().unwrap();
}

以上代码创建了一个名为users的表,并在事务内分别插入了两个元素,数据被成功提交到了MySQL数据库,我们看到了名为AliceBob的条目。

异步IO#

Rust语言具有异步IO处理的优势。在Rust中使用MySQL异步IO时,可以使用tokio-mysql crate来实现。tokio-mysql crate是一个基于Tokio实现的异步MySQL数据库客户端。

下面是使用tokio-mysql crate的示例代码:

use std::str::FromStr;
use tokio::runtime::Builder;
use tokio::time::Duration;
use tokio_mysql::{prelude::*, Error, Opts, Pool};
#[tokio::main]
async fn main() -> Result<(), Error> {
let opts = Opts::from_url("mysql://root:password@localhost:3306/test")?;
let pool = Pool::new(opts);
let pool = match pool {
Ok(p) => p,
Err(e) => return Err(e),
};
let mut conn = pool.get_conn().await?;
conn.query_drop("CREATE TABLE IF NOT EXISTS students (
id INT PRIMARY KEY NOT NULL,
name TEXT NOT NULL,
age INT NOT NULL
)")
.await?;
let id = 1;
let name = "Alice";
let age = 23;
conn.exec_drop(
format!(
"INSERT INTO students (id, name, age) VALUES ({}, \"{}\", {})",
id, name, age
)
.as_str()
)
.await?;
let mut conn2 = pool.get_conn().await?;
let result = conn2
.query_iter(
String::from("SELECT * FROM students")
.as_str(),
)
.await?;
for row in result {
let id: u32 = row.unwrap().take("id").unwrap().as_integer().unwrap().try_into().unwrap();
let name: &str = row.unwrap().take("name").unwrap().as_sql_str();
let age: u32 = row.unwrap().take("age").unwrap().as_integer().unwrap().try_into().unwrap();
println!("{} is {} years old", name, age);
}
Ok(())
}

它创建了一个名为students的表,并插入了一个名为Alice年龄为23的元素,然后遍历该表并打印结果。

Rust使用MySQL的最佳实践#

连接池#

在连接MySQL数据库时,使用连接池是非常重要的。连接池是一个预先创建的连接集合,由于预先初始化了这些连接,因此在保持连接上下文的情况下执行多个操作变得更加轻松。在Rust中使用mysql::Pool连接MySQL数据库是非常常见的。

let opts = OptsBuilder::new()
.ip_or_hostname(Some("localhost"))
.user(Some("root"))
.pass(Some("password"))
.db_name(Some("test"))
.tcp_port(3306);
let pool = Pool::new(opts).unwrap();

避免SQL注入#

避免SQL注入攻击是一个重要的安全问题。在Rust中使用mysql crate,可以使用mysql::from_valuemysql::Value::from方法来避免SQL注入攻击。

在Rust中,需要使用以下代码实现SQL语句中的参数绑定:

let name = "Alice";
let age = 23;
conn.prep_exec("INSERT INTO students (name, age) VALUES (?, ?)", (name, age),).unwrap();

在将参数传递给SQL查询时,需要使用mysql::Value::from方法将变量转换为mysql::Value类型,以防止SQL注入攻击。要从mysql::Value转换回常规变量,可以使用mysql::from_value方法。使用以下示例代码:

use mysql::*;
fn main() {
let result: Vec<Row> = conn.query("SELECT * FROM students WHERE age >= ?", (age.into(),)).unwrap();
for row in result {
let age: i32 = from_value(row.get("age").unwrap());
let name: String = from_value(row.get("name").unwrap());
println!("{} is {} years old", name, age);
}
}

SQL执行和结果处理#

在Rust中,可以使用mysql::Conn::querymysql::Conn::exec_itermysql::Conn::prep_exec等方法来执行SQL语句。但是,这些方法返回的结果类型有很大不同。query方法返回包含所有结果集的Vec<mysql::Row>类型,而exec_iter方法返回mysql::Row类型的迭代器。最后,prep_exec方法是最常用的方法,它可以绑定参数,并类似于通过命令行客户端发送的查询,并返回mysql::QueryResult类型。如果要提取单个结果,可以使用mysql::QueryResult的方法mysql::QueryResult::next来获取。

使用以下代码示例说明不同方法的使用:

use mysql::*;
fn main() {
let result: Vec<Row> = conn.query("SELECT * FROM students WHERE age >= ?", (age.into(),)).unwrap();
for row in result {
let age: i32 = from_value(row.get("age").unwrap());
let name: String = from_value(row.get("name").unwrap());
println!("{} is {} years old", name, age);
}
let mut iter = conn.exec_iter("SELECT age FROM students WHERE name = \"Alice\"").unwrap();
while let Some(result) = iter.next() {
let age: i32 = from_value(result.unwrap());
println!("Alice is {} years old", age);
}
conn.prep_exec("INSERT INTO students (name, age) VALUES (?, ?)",(name, age),).unwrap();
}

结论#

本教程介绍了如何在Rust中连接和使用MySQL数据库。我们学习了使用mysql crate连接MySQL数据库,并实现了一些常见用例,例如事务、异步IO等。此外,我们使用连接池、避免SQL注入技巧,并使用了不同的SQL执行和结果处理方法。如果你正在使用Rust编程语言,那么通过学习本教程,你将掌握基本和高级的连接和使用MySQL技巧。

支持与分享

如果这篇文章对你有帮助,欢迎分享给更多人或赞助支持!

赞助
Rust语言从入门到精通系列 - Rust语言MySQL实战
https://tinyzzh.github.io/posts/2023-03-24-rust_lang_tutorial_205_mysql/
作者
TinyZ Zzh
发布于
2023-03-24
许可协议
CC BY-NC-SA 4.0

评论区

Profile Image of the Author
TinyZ Zzh
专注于高并发服务器、网络游戏相关(Java、PHP、Unity3D、Unreal Engine等)技术,热爱游戏事业, 正在努力实现自我价值当中。
公告
欢迎来到我的博客!这是一则示例公告。
音乐
封面

音乐

暂未播放

0:00 0:00
暂无歌词
分类
标签
站点统计
文章
211
分类
38
标签
200
总字数
337,853
运行时长
0
最后活动
0 天前

文章目录