数据库操作与数据管理——Rust 与 SQLite 的集成
第六章:数据库操作与数据管理
第一节:Rust 与 SQLite 的集成
在本节中,我们将深入探讨如何在 Rust 中使用 SQLite 数据库,涵盖从基本的 CRUD 操作到事务处理、数据模型的构建、性能优化以及安全性考虑等方面。SQLite 是一个轻量级的关系型数据库,适合嵌入式应用和小型项目。我们将利用 rusqlite 库高效地与 SQLite 进行交互。
1. 使用 rusqlite 进行基本 CRUD 操作
1.1 rusqlite 库的引入
首先,在 Cargo.toml 文件中添加 rusqlite 依赖:
[dependencies]
rusqlite = { version = "0.26", features = ["bundled"] }
1.2 连接到 SQLite 数据库
use rusqlite::{params, Connection, Result};fn connect_to_db() -> Result<Connection> {let conn = Connection::open("my_database.db")?;Ok(conn)
}
1.3 创建表
在执行任何 CRUD 操作之前,我们需要定义数据表。
fn create_table(conn: &Connection) -> Result<()> {conn.execute("CREATE TABLE IF NOT EXISTS users (id INTEGER PRIMARY KEY,name TEXT NOT NULL,age INTEGER NOT NULL)",[],)?;Ok(())
}
1.4 插入数据(Create)
我们可以通过以下代码插入数据到表中:
fn insert_user(conn: &Connection, name: &str, age: i32) -> Result<()> {conn.execute("INSERT INTO users (name, age) VALUES (?1, ?2)",params![name, age],)?;Ok(())
}
1.5 查询数据(Read)
查询数据可以使用 query_map 方法:
fn fetch_users(conn: &Connection) -> Result<Vec<(i32, String, i32)>> {let mut stmt = conn.prepare("SELECT id, name, age FROM users")?;let user_iter = stmt.query_map([], |row| {Ok((row.get(0)?, row.get(1)?, row.get(2)?))})?;let mut users = Vec::new();for user in user_iter {users.push(user?);}Ok(users)
}
1.6 更新数据(Update)
fn update_user_age(conn: &Connection, id: i32, new_age: i32) -> Result<()> {conn.execute("UPDATE users SET age = ?1 WHERE id = ?2",params![new_age, id],)?;Ok(())
}
1.7 删除数据(Delete)
fn delete_user(conn: &Connection, id: i32) -> Result<()> {conn.execute("DELETE FROM users WHERE id = ?1", params![id])?;Ok(())
}
2. 处理事务与连接池
2.1 事务的基本操作
使用事务可以确保一系列数据库操作的原子性。
fn transaction_example(conn: &Connection) -> Result<()> {conn.execute("BEGIN TRANSACTION", [])?;insert_user(conn, "Alice", 30)?;insert_user(conn, "Bob", 25)?;conn.execute("COMMIT", [])?;Ok(())
}
2.2 处理错误与回滚
在执行事务时,如果出现错误,我们应该回滚事务。
fn safe_transaction(conn: &Connection) -> Result<()> {let transaction = conn.transaction()?;transaction.execute("INSERT INTO users (name, age) VALUES (?1, ?2)", params!["Charlie", 28])?;// 故意造成错误transaction.execute("INSERT INTO users (name, age) VALUES (?1, ?2)", params![None::<String>, 28])?;transaction.commit()?;Ok(())
}
2.3 使用连接池
使用连接池可以提高性能,允许多个线程安全地共享数据库连接。
use r2d2::Pool;
use r2d2_sqlite::SqliteConnectionManager;fn create_pool() -> Pool<SqliteConnectionManager> {let manager = SqliteConnectionManager::new("my_database.db");Pool::builder().build(manager).unwrap()
}
3. 数据模型与查询构建
3.1 定义数据模型
使用 Rust 结构体表示数据模型,使数据操作更加明确。
#[derive(Debug)]
struct User {id: i32,name: String,age: i32,
}
3.2 从数据库映射到模型
编写函数将数据库记录映射到模型:
fn map_row_to_user(row: &rusqlite::Row) -> User {User {id: row.get(0).unwrap(),name: row.get(1).unwrap(),age: row.get(2).unwrap(),}
}
3.3 动态查询构建
使用 rusqlite 的 Query 来构建动态查询,以支持不同的查询条件。
fn fetch_users_with_conditions(conn: &Connection, min_age: Option<i32>) -> Result<Vec<User>> {let mut query = String::from("SELECT id, name, age FROM users");let mut params: Vec<&(dyn rusqlite::ToSql + 'static)> = Vec::new();if let Some(age) = min_age {query.push_str(" WHERE age >= ?");params.push(&age);}let mut stmt = conn.prepare(&query)?;let user_iter = stmt.query_map(params, |row| map_row_to_user(row))?;let mut users = Vec::new();for user in user_iter {users.push(user?);}Ok(users)
}
3.4 连接模型与业务逻辑
通过将数据库操作封装在服务层,将数据模型与业务逻辑分离,确保代码的可维护性。
struct UserService {conn: Connection,
}impl UserService {fn new(conn: Connection) -> Self {UserService { conn }}fn add_user(&self, name: &str, age: i32) -> Result<()> {insert_user(&self.conn, name, age)}fn get_all_users(&self) -> Result<Vec<User>> {fetch_users(&self.conn)}
}
4. 性能优化
4.1 使用索引
在处理大量数据时,使用索引可以显著提高查询速度。
fn create_index(conn: &Connection) -> Result<()> {conn.execute("CREATE INDEX idx_user_name ON users (name)", [])?;Ok(())
}
4.2 批量插入
在插入大量数据时,可以使用批量插入来提高性能。
fn batch_insert(conn: &Connection, users: Vec<(String, i32)>) -> Result<()> {let tx = conn.transaction()?;for (name, age) in users {tx.execute("INSERT INTO users (name, age) VALUES (?1, ?2)", params![name, age])?;}tx.commit()?;Ok(())
}
4.3 减少数据库往返
尽量减少与数据库的交互次数,可以通过使用预处理语句或批量操作实现。
5. 安全性考虑
在进行数据库操作时,确保数据的安全性和完整性是至关重要的。以下是一些重要的安全性考虑因素,帮助开发者在使用 Rust 与 SQLite 进行数据库操作时,最大程度地保护应用程序和用户数据。
5.1 防止 SQL 注入
SQL 注入是数据库安全中最常见的攻击方式之一。攻击者通过在 SQL 查询中插入恶意代码,从而获取、修改或删除数据。为了防止 SQL 注入,采用参数化查询是最有效的方法。
fn safe_insert_user(conn: &Connection, name: &str, age: i32) -> Result<()> {conn.execute("INSERT INTO users (name, age) VALUES (?1, ?2)",params![name, age],)?;Ok(())
}
关键点:
- 参数化查询:总是使用参数化查询而不是将用户输入直接拼接到 SQL 字符串中。
- 使用库功能:利用
rusqlite提供的安全 API,避免手动构建 SQL 字符串。
5.2 数据加密
对于敏感数据,尤其是用户的个人信息、信用卡号码等,应该使用加密技术进行存储。Rust 提供了多个加密库,比如 aes、rust-crypto 等,可以轻松实现数据的加密和解密。
use aes::{Aes128, NewBlockCipher, BlockEncrypt, BlockDecrypt};
use aes::block_cipher_trait::generic_array::GenericArray;fn encrypt_data(data: &[u8], key: &[u8; 16]) -> Vec<u8> {let cipher = Aes128::new(GenericArray::from_slice(key));let mut block = GenericArray::clone_from_slice(data);cipher.encrypt_block(&mut block);block.to_vec()
}fn decrypt_data(encrypted_data: &[u8], key: &[u8; 16]) -> Vec<u8> {let cipher = Aes128::new(GenericArray::from_slice(key));let mut block = GenericArray::clone_from_slice(encrypted_data);cipher.decrypt_block(&mut block);block.to_vec()
}
关键点:
- 选择合适的加密算法:选择适合您应用程序需求的加密算法,AES 是一个普遍推荐的选择。
- 密钥管理:确保密钥的安全存储与管理,避免将密钥硬编码到代码中。
5.3 用户输入验证
确保所有用户输入都经过严格验证,避免不合法或恶意数据进入数据库。这包括检查数据类型、长度、格式等。
fn validate_user_input(name: &str, age: i32) -> Result<()> {if name.len() > 50 {return Err(rusqlite::Error::UserFunctionError("Name too long".into()));}if age < 0 {return Err(rusqlite::Error::UserFunctionError("Invalid age".into()));}Ok(())
}
关键点:
- 使用正则表达式:对复杂输入(如电子邮件、电话号码)使用正则表达式进行验证。
- 长度和类型检查:确保输入的数据类型和长度符合预期,防止意外错误和数据损坏。
5.4 身份验证与授权
确保只有经过身份验证的用户才能访问敏感数据或执行重要操作。可以使用 JWT(JSON Web Tokens)或 OAuth 2.0 等现代身份验证方法。
fn verify_user(token: &str) -> Result<bool> {// 此处可使用 JWT 解码和验证逻辑Ok(true) // 简化示例
}
关键点:
- 确保安全的认证流程:实现安全的登录流程,使用 HTTPS 保护数据传输。
- 最小权限原则:每个用户和角色应仅能访问其工作所需的数据,避免不必要的权限。
5.5 日志记录与监控
有效的日志记录和监控可以帮助识别和响应安全事件。记录所有的数据库操作、异常和潜在的安全威胁。
fn log_operation(action: &str) {// 记录数据库操作println!("Operation logged: {}", action);
}
关键点:
- 详细的日志记录:记录操作的时间、执行者和操作内容,便于后续审计。
- 监控与警报:设置监控系统,及时警报异常活动,例如频繁的失败登录尝试。
小结
在数据库操作中,安全性是一个不容忽视的重要方面。通过防止 SQL 注入、加密敏感数据、验证用户输入、实施身份验证和授权机制以及有效的日志记录和监控,可以显著提高应用程序的安全性。务必在开发过程中将这些安全性考虑融入设计中,以保护用户数据和应用程序的完整性。
进一步学习
- 深入研究 Rust 的安全特性:了解 Rust 语言如何在编译时避免许多常见的安全漏洞。
- 数据库安全最佳实践:研究数据库安全的行业标准和最佳实践。
- 加密算法的选择与实现:探索不同加密算法的优缺点,并尝试在 Rust 中实现更多的加密技术。

相关文章:
数据库操作与数据管理——Rust 与 SQLite 的集成
第六章:数据库操作与数据管理 第一节:Rust 与 SQLite 的集成 在本节中,我们将深入探讨如何在 Rust 中使用 SQLite 数据库,涵盖从基本的 CRUD 操作到事务处理、数据模型的构建、性能优化以及安全性考虑等方面。SQLite 是一个轻量…...
LeetCode 0063.不同路径 II:动态规划 - 原地使用地图数组,几乎无额外空间开销
【LetMeFly】63.不同路径 II:动态规划 - 原地使用地图数组,几乎无额外空间开销 力扣题目链接:https://leetcode.cn/problems/unique-paths-ii/ 给定一个 m x n 的整数数组 grid。一个机器人初始位于 左上角(即 grid[0][0]&#…...
elementui:el-table支持搜索、切换分页多选功能,以及数据回显
1、el-table相关代码,需注意:row-key"(row) > { return row.id }" 以及 :reserve-selection"true" <div class"boxList"><div class"search-form"><!-- 搜索表单 --><el-form :inline"true&q…...
深度整理总结MySQL——索引正确使用姿势
索引正确使用姿势 前言MySQL索引优缺点分析✅ 索引的优势⚠️ 索引的代价 如何合理建立索引?——关键原则总结重要的优化机制索引覆盖——通俗的方式讲解索引下推索引跳跃式扫描 前言 这篇文章是补充一些基本概念和实战的一些使用建议. MySQL索引优缺点分析 ✅ 索引的优势 …...
使用LLaMA Factory踩坑记录
前置条件:电脑显卡RTX 4080 问题:LLaMA-Factory在运行的时候,弹出未检测到CUDA的报错信息 结论:出现了以上的报错,主要可以归结于以下两个方面: 1、没有安装GPU版本的pytorch,下载的是CPU版本…...
亚博microros小车-原生ubuntu支持系列:25 二维码控制运动
二维码识别 安装依赖 pip3 install pyzbarsudo apt install libzbar-dev 在用小车识别之前,先用电脑的摄像头测试下基本的识别 import cv2 import rclpy from rclpy.node import Node import pyzbar.pyzbar as pyzbar import numpy as np from ament_index_pyth…...
基于深度学习的人工智能量化衰老模型构建与全流程应用研究
一、引言 1.1 研究背景与意义 1.1.1 人口老龄化现状与挑战 人口老龄化是当今全球面临的重要社会趋势之一,其发展态势迅猛且影响深远。根据联合国的相关数据,1980 年,全球 65 岁及以上人口数量仅为 2.6 亿,到 2021 年,这一数字已翻番,达到 7.61 亿,而预计到 2050 年,…...
【医院运营统计专题】2.运营统计:医院管理的“智慧大脑”
医院成本核算、绩效管理、运营统计、内部控制、管理会计专题索引 引言 在当今医疗行业快速发展的背景下,医院运营管理的科学性和有效性成为了决定医院竞争力和可持续发展能力的关键因素。运营统计作为医院管理的重要工具,通过对医院各类数据的收集、整理、分析和解读,为医…...
Spring Boot Actuator使用
说明:本文介绍Spring Boot Actuator的使用,关于Spring Boot Actuator介绍,下面这篇博客写得很好,珠玉在前,我就不多介绍了。 Spring Boot Actuator 简单使用 项目里引入下面这个依赖 <!--Spring Boot Actuator依…...
【AI应用】免费的文本转语音工具:微软 Edge TTS 和 开源版 ChatTTS 对比
【AI论文解读】【AI知识点】【AI小项目】【AI战略思考】【AI日记】【读书与思考】【AI应用】 我试用了下Edge TTS,感觉还不错,不过它不支持克隆声音(比如自己的声音) 微软 Edge TTS 和 开源版 ChatTTS 都是免费的 文本转语音&…...
如何在 Qt 中添加和使用系统托盘图标
在 Qt 中实现系统托盘图标是一个常见的需求,尤其是在桌面应用程序中。系统托盘图标可以让应用程序在后台运行时仍然具有可见性,同时避免占用过多的桌面空间。本文将详细介绍如何在 Qt 项目中添加托盘图标,并通过资源系统(.qrc 文件…...
【WB 深度学习实验管理】利用 Hugging Face 实现高效的自然语言处理实验跟踪与可视化
本文使用到的 Jupyter Notebook 可在GitHub仓库002文件夹找到,别忘了给仓库点个小心心~~~ https://github.com/LFF8888/FF-Studio-Resources 在自然语言处理领域,使用Hugging Face的Transformers库进行模型训练已经成为主流。然而,随着模型复…...
基础入门-网站协议身份鉴权OAuth2安全Token令牌JWT值Authirization标头
知识点: 1、网站协议-http/https安全差异(抓包) 2、身份鉴权-HTTP头&OAuth2&JWT&Token 一、演示案例-网站协议-http&https-安全测试差异性 1、加密方式 HTTP:使用明文传输,数据在传输过程中可以被…...
C语言基础系列【3】VSCode使用
前面我们提到过VSCode有多么的好用,本文主要介绍如何使用VSCode编译运行C语言代码。 安装 首先去官网(https://code.visualstudio.com/)下载安装包,点击Download for Windows 获取安装包后,一路点击Next就可以。 配…...
MySQL-5.7.44安装(CentOS7)
目录 1、下载安装包并解压 2、创建数据目录与日志目录 3、设置环境变量 4、刷新环境变量 5、执行初始化 6、创建配置文件目录 7、新建配置文件 8、为安装目录赋予可执行权限 9、创建服务启动脚本 10、启动服务并将启动脚本加入开机自启动 11、查看服务状态 12、创建…...
服务端与多客户端照片的传输,recv,send
一、照片传输 server.c /* * 文件名称:server.c * 创 建 者: * 创建日期:2025年02月07日 * 描 述: */ #include <stdio.h> #include <sys/types.h> /* See NOTES */ #include <sys/socket.h…...
JS实现灯光闪烁效果
在 JS中,我们可以实现灯光闪烁效果,这里主要用 setInterval 和 clearInterval 两个重要方法。 效果图 源代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>灯闪烁效果<…...
SpringCloud面试题----Nacos和Eureka的区别
功能特性 服务发现 Nacos:支持基于 DNS 和 RPC 的服务发现,提供了更为灵活的服务发现机制,能满足不同场景下的服务发现需求。Eureka:主要基于 HTTP 的 RESTful 接口进行服务发现,客户端通过向 Eureka Server 发送 HT…...
verilog练习:i2c slave 模块设计
文章目录 前言1. 结构2.代码2.1 iic_slave.v2.2 sync.v2.3 wr_fsm.v2.3.1 状态机状态解释 2.4 ram.v 3. 波形展示4. 建议5. 资料总结 前言 首先就不啰嗦iic协议了,网上有不少资料都是叙述此协议的。 下面将是我本次设计的一些局部设计汇总,如果对读者有…...
3.5 Go(特殊函数)
目录 一、匿名函数 1、匿名函数的特点: 2、匿名函数代码示例 2、匿名函数的类型 二、递归函数 1. 递推公式版本 2. 循环改递归 三、嵌套函数 1、嵌套函数用途 2、代码示例 3、作用域 & 变量生存周期 四、闭包 1、闭包使用场景 2、代码示例 五、De…...
从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...
Map相关知识
数据结构 二叉树 二叉树,顾名思义,每个节点最多有两个“叉”,也就是两个子节点,分别是左子 节点和右子节点。不过,二叉树并不要求每个节点都有两个子节点,有的节点只 有左子节点,有的节点只有…...
是否存在路径(FIFOBB算法)
题目描述 一个具有 n 个顶点e条边的无向图,该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序,确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数,分别表示n 和 e 的值(1…...
服务器--宝塔命令
一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行! sudo su - 1. CentOS 系统: yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...
Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习)
Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习) 一、Aspose.PDF 简介二、说明(⚠️仅供学习与研究使用)三、技术流程总览四、准备工作1. 下载 Jar 包2. Maven 项目依赖配置 五、字节码修改实现代码&#…...
Java + Spring Boot + Mybatis 实现批量插入
在 Java 中使用 Spring Boot 和 MyBatis 实现批量插入可以通过以下步骤完成。这里提供两种常用方法:使用 MyBatis 的 <foreach> 标签和批处理模式(ExecutorType.BATCH)。 方法一:使用 XML 的 <foreach> 标签ÿ…...
探索Selenium:自动化测试的神奇钥匙
目录 一、Selenium 是什么1.1 定义与概念1.2 发展历程1.3 功能概述 二、Selenium 工作原理剖析2.1 架构组成2.2 工作流程2.3 通信机制 三、Selenium 的优势3.1 跨浏览器与平台支持3.2 丰富的语言支持3.3 强大的社区支持 四、Selenium 的应用场景4.1 Web 应用自动化测试4.2 数据…...
Bean 作用域有哪些?如何答出技术深度?
导语: Spring 面试绕不开 Bean 的作用域问题,这是面试官考察候选人对 Spring 框架理解深度的常见方式。本文将围绕“Spring 中的 Bean 作用域”展开,结合典型面试题及实战场景,帮你厘清重点,打破模板式回答,…...
MyBatis中关于缓存的理解
MyBatis缓存 MyBatis系统当中默认定义两级缓存:一级缓存、二级缓存 默认情况下,只有一级缓存开启(sqlSession级别的缓存)二级缓存需要手动开启配置,需要局域namespace级别的缓存 一级缓存(本地缓存&#…...
沙箱虚拟化技术虚拟机容器之间的关系详解
问题 沙箱、虚拟化、容器三者分开一一介绍的话我知道他们各自都是什么东西,但是如果把三者放在一起,它们之间到底什么关系?又有什么联系呢?我不是很明白!!! 就比如说: 沙箱&#…...
