ORM++ 封装实战指南:安全高效的 C++ MySQL 数据库操作

ORM++ 封装实战指南:安全高效的 C++ MySQL 数据库操作

一、环境准备

1.1 依赖安装

# Ubuntu/Debian
sudo apt-get install libmysqlclient-dev
# CentOS
sudo yum install mysql-devel# 编译时链接库 (-I 指定头文件路径 -L 指定库路径)
g++ main.cpp -std=c++17 -I/usr/include/mysql -lmysqlclient -lpthread

1.2 ORM++ 集成

// 下载头文件版 ORM++
#include <ormpp/mysql.hpp>
#include <ormpp/dbng.hpp>

二、模型定义规范

2.1 数据表映射

#pragma once
#include <string>
#include <ormpp/ormpp.hpp>struct User {int id;        // 必须包含主键字段std::string username;std::string email;time_t create_time;
};
REFLECTION(User, id, username, email, create_time);// 带注释的进阶版
struct Article {int article_id;  // 主键std::string title;std::string content;int author_id;   // 外键ormpp::date create_date;
};
REFLECTION_WITH_NAME(Article, "articles", ormpp::field_constraint{&Article::article_id, "PRIMARY KEY AUTO_INCREMENT"},ormpp::field_constraint{&Article::title, "VARCHAR(255) NOT NULL"},ormpp::field_constraint{&Article::author_id, "REFERENCES users(id)"}
);

三、核心封装类设计

3.1 数据库管理器

class DatabaseManager {
public:// 单例模式获取实例static DatabaseManager& Instance() {static DatabaseManager instance;return instance;}// 初始化连接池bool Initialize(const std::string& host,const std::string& user,const std::string& password,const std::string& db,int port = 3306,int poolSize = 5);// 获取连接(RAII 管理)ormpp::dbng<ormpp::mysql> GetConnection();// CRUD 操作封装template <typename T, typename... WhereArgs>std::vector<T> Query(const std::string& condition = "", WhereArgs&&... args);template <typename T>int Insert(const T& entity);template <typename T, typename... UpdateArgs>int Update(const T& entity, const std::string& condition, UpdateArgs&&... args);template <typename... DeleteArgs>int Delete(const std::string& condition, DeleteArgs&&... args);private:std::queue<ormpp::dbng<ormpp::mysql>> connectionPool_;std::mutex poolMutex_;std::condition_variable poolCV_;
};

四、CRUD 高级封装

4.1 安全查询

template <typename T, typename... Args>
std::vector<T> DatabaseManager::Query(const std::string& condition, Args&&... args) {auto conn = GetConnection();try {// 自动生成 SELECT 语句std::string sql = ormpp::generate_query_sql<T>(condition);// 执行参数化查询auto result = conn->template query<T>(sql, std::forward<Args>(args)...);ZRY_LOG_DEBUG("Query returned {} records", result.size());return result;} catch (const std::exception& e) {ZRY_LOG_ERROR("Query failed: {}", e.what());throw;}
}// 使用示例
auto users = DatabaseManager::Instance().Query<User>("WHERE age > ?", 18);

4.2 智能插入

template <typename T>
int DatabaseManager::Insert(const T& entity) {auto conn = GetConnection();try {int affected = conn->insert(entity);if (affected > 0) {ZRY_LOG_INFO("Inserted {} successfully", typeid(T).name());}return affected;} catch (const ormpp::mysql_exception& e) {ZRY_LOG_ERROR("Insert failed: {}", e.what());return -1;}
}// 支持批量插入
template <>
int DatabaseManager::Insert<std::vector<User>>(const std::vector<User>& users) {// ... 批量插入优化实现
}

五、事务管理

5.1 事务封装类

class Transaction {
public:explicit Transaction(DatabaseManager& dbMgr) : dbMgr_(dbMgr), conn_(dbMgr.GetConnection()) {conn_->execute("START TRANSACTION");}~Transaction() {if (!committed_) {conn_->execute("ROLLBACK");}}bool Commit() {try {conn_->execute("COMMIT");committed_ = true;return true;} catch (...) {return false;}}private:DatabaseManager& dbMgr_;ormpp::dbng<ormpp::mysql> conn_;bool committed_ = false;
};// 使用示例
try {Transaction trans(DatabaseManager::Instance());// 执行多个操作trans.Commit();
} catch (...) {// 自动回滚
}

六、查询构建器

6.1 链式调用封装

class QueryBuilder {
public:QueryBuilder& Where(const std::string& condition) {whereClause_ = "WHERE " + condition;return *this;}QueryBuilder& OrderBy(const std::string& field, bool asc = true) {orderByClause_ = "ORDER BY " + field + (asc ? " ASC" : " DESC");return *this;}QueryBuilder& Limit(int count, int offset = 0) {limitClause_ = "LIMIT " + std::to_string(offset) + "," + std::to_string(count);return *this;}template <typename T>std::vector<T> Execute() {std::string sql = ormpp::generate_query_sql<T>("")+ " " + whereClause_+ " " + orderByClause_+ " " + limitClause_;return DatabaseManager::Instance().Query<T>(sql);}private:std::string whereClause_;std::string orderByClause_;std::string limitClause_;
};// 使用示例
auto users = QueryBuilder().Where("age > ? AND status = ?").OrderBy("create_time", false).Limit(10).Execute<User>();

七、性能优化

7.1 连接池实现要点

bool DatabaseManager::Initialize(...) {for (int i = 0; i < poolSize; ++i) {auto conn = std::make_shared<ormpp::dbng<ormpp::mysql>>();if (conn->connect(host, user, password, db, port)) {connectionPool_.push(conn);}}return !connectionPool_.empty();
}ormpp::dbng<ormpp::mysql> DatabaseManager::GetConnection() {std::unique_lock<std::mutex> lock(poolMutex_);poolCV_.wait(lock, [this]{ return !connectionPool_.empty(); });auto conn = connectionPool_.front();connectionPool_.pop();return conn;
}void ReleaseConnection(ormpp::dbng<ormpp::mysql> conn) {std::lock_guard<std::mutex> lock(poolMutex_);connectionPool_.push(conn);poolCV_.notify_one();
}

八、日志与监控

8.1 SQL 执行日志

// 在查询执行前记录
ZRY_LOG_DEBUG("Executing SQL: {}", sql);// 执行后统计耗时
auto start = std::chrono::steady_clock::now();
// ... execute query ...
auto end = std::chrono::steady_clock::now();
ZRY_LOG_PERF("Query took {}ms", std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count());

九、最佳实践

  1. 参数化查询​:始终使用 ? 占位符,防止 SQL 注入
db.Query<User>("WHERE username = ?", "john_doe");
  1. 类型安全​:利用 ORM 的自动类型转换
auto products = db.Query<Product>("WHERE price > ?", 100.0f);
  1. 连接生命周期​:使用 RAII 管理连接
{auto conn = db.GetConnection();// 使用连接...
} // 自动释放回连接池
  1. 错误处理​:统一异常捕获策略
try {// 数据库操作
} catch (const ormpp::mysql_exception& e) {// 处理数据库特定错误
} catch (const std::exception& e) {// 处理通用错误
}

十、扩展方向

  1. 异步操作​:结合 libuv 或 Boost.Asio 实现异步查询
  2. 二级缓存​:集成 Redis 缓存高频查询结果
  3. 数据分片​:实现跨多个数据库实例的 Sharding
  4. 健康检查​:定期验证连接有效性
  5. SQL 分析​:记录慢查询日志,优化索引

通过以上封装,开发者可以:
✅ 减少 70% 以上的样板代码
✅ 提升 SQL 注入防护等级
✅ 统一管理数据库访问策略
✅ 方便进行性能监控和优化
✅ 快速适应数据库 schema 变更

完整示例代码参见:[GitHub 仓库链接]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/diannao/84541.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JESD204B 协议介绍

一、协议概述 JESD204B是由JEDEC&#xff08;固态技术协会&#xff09;制定的高速串行接口标准&#xff0c;专为模数转换器&#xff08;ADC&#xff09;、数模转换器&#xff08;DAC&#xff09;与逻辑器件&#xff08;如FPGA、ASIC&#xff09;之间的数据传输设计。其核心目标…

yolov8,c++案例汇总

文章目录 引言多目标追踪案例人体姿态估计算法手势姿态估计算法目标分割算法 引言 以下案例,基于c,ncnn,yolov8既可以在windows10/11上部署, 也可以在安卓端部署, 也可以在嵌入式端部署, 服务器端可支持部署封装为DLL,支持c/c#/java端调用 多目标追踪案例 基于yolov8, ncnn,…

运动规划实战案例 | 图解基于状态晶格(State Lattice)的路径规划(附ROS C++/Python仿真)

目录 1 控制采样 vs 状态采样2 State Lattice路径规划2.1 算法流程2.2 Lattice运动基元生成2.3 几何代价函数2.4 运动学约束启发式 3 算法仿真3.1 ROS C仿真3.2 Python仿真 1 控制采样 vs 状态采样 控制采样的技术路线源自经典的运动学建模思想。这种方法将机器人的控制指令空…

BERT框架:自然语言处理的革命性突破

引言 在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;2018年Google推出的BERT&#xff08;Bidirectional Encoder Representations from Transformers&#xff09;框架无疑是一场革命。作为基于Transformer架构的双向编码器表示模型&#xff0c;BERT通过预训练学习…

【Fifty Project - D31】

结束了一个超级消耗周末&#xff0c;满安排之健身梅溪湖游泳做饭喝酒羽毛球赛 完全力竭了&#xff0c;久久不能恢复过来&#xff0c;暂停健身安排了 端午后再继续 今日完成记录 TimePlan完成情况7&#xff1a;30 - 8&#xff1a;10有氧爬坡√9&#xff1a;00 - 11&#xff1a;…

信息学奥赛一本通 1547:【 例 1】区间和

【题目链接】 ybt 1547&#xff1a;【 例 1】区间和 【题目考点】 1. 线段树 2. 树状数组 【解题思路】 本题要求维护区间和&#xff0c;实现单点修改、区间查询。 解法1&#xff1a;线段树 线段树原理&#xff0c;及实现方法见&#xff1a;洛谷 P3374 【模板】树状数组…

力扣面试150题--求根节点到叶节点数字之和

Day 48 题目描述 思路 我们利用sum这个全局变量来保存总和值&#xff0c;递归函数sum来计算每个根到叶子节点路径所代表的数&#xff0c;由于我们需要遍历到每条根到叶子节点的路径&#xff0c;所有我采取了前序遍历&#xff0c;如果不是叶子节点&#xff0c;就计算到该节点代…

DJI上云API官方demo学习

1、websocket&#xff0c;所在位置如下图&#xff0c;调用的可以用//websocket搜索 2、用到的http客户端&#xff0c;axios 3、很多和后端交互都是走的http请求

uniapp开发小程序,如何根据权限动态配置按钮或页面内容

前言 写了好几个项目&#xff0c;发现小程序对权限控制非常麻烦&#xff0c;于是有了这个想法&#xff0c;但是网上找了一圈没有一个比较完善的讲解&#xff0c;因为小程序不支持自定义指令&#xff0c;所以不能像后台那样方便&#xff0c;于是就将几个博主的想法结合。 思路就…

LSTM+Transformer混合模型架构文档

LSTMTransformer混合模型架构文档 模型概述 本项目实现了一个LSTMTransformer混合模型&#xff0c;用于超临界机组协调控制系统的数据驱动建模。该模型结合了LSTM的时序建模能力和Transformer的自注意力机制&#xff0c;能够有效捕捉时间序列数据中的长期依赖关系和变量间的复…

测量尺子:多功能测量工具,科技改变生活

测量尺子是一款专业的测距仪测量万能工具箱类型手机APP&#xff0c;旨在为用户提供最贴心的测量助手。它拥有和现实测量仪器一样的测量标准&#xff0c;更简单便捷且精准的测量方式&#xff0c;最新AR科技测量更是大大拓宽了可以被测量的高度和深度。无论是日常使用、学习还是工…

结课作业01. 用户空间 MPU6050 体感鼠标驱动程序

目录 一. qt界面实现 二. 虚拟设备模拟模拟鼠标实现体感鼠标 2.1 函数声明 2.2 虚拟鼠标实现 2.2.1 虚拟鼠标创建函数 2.2.2 鼠标移动函数 2.2.3 鼠标点击函数 2.3 mpu6050相关函数实现 2.3.1 i2c设备初始化 2.3.2 mpu6050寄存器写入 2.3.3 mpu6050寄存器读取 2.3.…

深入浅出 Python Testcontainers:用容器优雅地编写集成测试

在现代软件开发中&#xff0c;自动化测试已成为敏捷开发与持续集成中的关键环节。单元测试可以快速验证函数或类的行为是否符合预期&#xff0c;而集成测试则确保多个模块协同工作时依然正确。问题是&#xff1a;如何让集成测试可靠、可重复且易于维护&#xff1f; 这时&#…

JVM 的垃圾回收器

新生代回收器 通性 会触发StW&#xff0c;暂停所有应用线程复制算法 Serial 单线程回收适合单线程系统 ParNew 多线程回收优先保证响应速度&#xff0c;降低 STW&#xff08;STW 越大&#xff0c;执行垃圾回收的时间越长&#xff0c;回收的垃圾越多&#xff0c;减少垃圾回…

【笔记】排查并解决Error in LLM call after 3 attempts: (status code: 502)

#工作记录 一、问题描述 在部署运行部署对冲基金分析工具 ai-hedge-fund 时&#xff0c;不断出现以下报错&#xff0c;导致项目运行异常&#xff1a; Error in LLM call after 3 attempts: (status code: 502) Error in LLM call after 3 attempts: [WinError 10054] 远程主…

GO 语言进阶之 Template 模板使用

更多个人笔记见&#xff1a; github个人笔记仓库 gitee 个人笔记仓库 个人学习&#xff0c;学习过程中还会不断补充&#xff5e; &#xff08;后续会更新在github上&#xff09; 文章目录 Template 模板基本示例语法1. 基本输出语法2. 控制结构3. 空白字符控制4. Must函数 Temp…

origin绘图之【如何将多条重叠、高度重叠的点线图、折线图分开】

在日常的数据可视化工作中&#xff0c;Origin 作为一款功能强大的科研绘图软件&#xff0c;广泛应用于实验数据处理、结果展示与论文图表制作等领域。然而&#xff0c;在处理多组数据、特别是绘制多条曲线的折线图或点线图时&#xff0c;常常会遇到这样一个困扰&#xff1a;多条…

Java基础 Day19

一、泛型&#xff08;JDK5引入&#xff09; 1、基本概念 在编译阶段约束操作的数据类型&#xff0c;并进行检查 好处&#xff1a;统一数据类型&#xff0c;将运行期的错误提升到了编译期 泛型的默认类型是 Object 2、泛型类 在创建类的时候写上泛型 在创建具体对象的时候…

Gitlab-Runner安装

文章目录 helm方式安装在K8S上参考gitlab CI/CD 文件变量缓存服务器K8S部署 docker镜像mavendocker安装docker buildx minionodehelmkubectlsonar-scanner-cli 问题清除cachehelm执行时无权限 下载镜像失败下载gitlab-runner镜像失败 Gitlab-ci中使用java前端 helm方式安装在K8…

在 Ubuntu linux系统中设置时区的方案

查看时区 在 Ubuntu 系统中&#xff0c;可以通过以下方法查看当前时区设置&#xff1a; 1. 使用 timedatectl 命令&#xff08;推荐&#xff09; 在终端运行以下命令&#xff1a; timedatectl输出示例&#xff1a; Local time: Sun 2025-05-25 10:30:00 CST Universal t…