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

news2025/6/1 7:31:06

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.coloradmin.cn/o/2392292.html

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

相关文章

kafka学习笔记(三、消费者Consumer使用教程——从指定位置消费)

1.简介 Kafka的poll()方法消费无法精准的掌握其消费的起始位置&#xff0c;auto.offset.reset参数也只能在比较粗粒度的指定消费方式。更细粒度的消费方式kafka提供了seek()方法可以指定位移消费允许消费者从特定位置&#xff08;如固定偏移量、时间戳或分区首尾&#xff09;开…

【后端高阶面经:架构篇】46、分布式架构:如何应对高并发的用户请求

一、架构设计原则:构建可扩展的系统基石 在分布式系统中,高并发场景对架构设计提出了极高要求。 分层解耦与模块化是应对复杂业务的核心策略,通过将系统划分为客户端、CDN/边缘节点、API网关、微服务集群、缓存层和数据库层等多个层次,实现各模块的独立演进与维护。 1.1 …

网络编程学习笔记——TCP网络编程

文章目录 1、socket()函数2、bind()函数3、listen()4、accept()5、connect()6、send()/write()7、recv()/read()8、套接字的关闭9、TCP循环服务器模型10、TCP多线程服务器11、TCP多进程并发服务器 网络编程常用函数 socket() 创建套接字bind() 绑定本机地址和端口connect() …

Vue+element-ui,实现表格渲染缩略图,鼠标悬浮缩略图放大,点击缩略图播放视频(一)

Vueelement-ui&#xff0c;实现表格渲染缩略图&#xff0c;鼠标悬浮缩略图放大&#xff0c;点击缩略图播放视频 前言整体代码预览图具体分析基础结构主要标签作用videoel-popover 前言 如标题&#xff0c;需要实现这样的业务 此处文章所实现的&#xff0c;是静态视频资源。 注…

day13 leetcode-hot100-22(链表1)

160. 相交链表 - 力扣&#xff08;LeetCode&#xff09; 1.哈希集合HashSet 思路 &#xff08;1&#xff09;将A链的所有数据存储到HashSet中。 &#xff08;2&#xff09;遍历B链&#xff0c;找到是否在A中存在。 具体代码 /*** Definition for singly-linked list.* pu…

【Oracle】DQL语言

个人主页&#xff1a;Guiat 归属专栏&#xff1a;Oracle 文章目录 1. DQL概述1.1 什么是DQL&#xff1f;1.2 DQL的核心功能 2. SELECT语句基础2.1 基本语法结构2.2 最简单的查询2.3 DISTINCT去重 3. WHERE条件筛选3.1 基本条件运算符3.2 逻辑运算符组合3.3 高级条件筛选 4. 排序…

HUAWEI华为MateBook D 14 2021款i5,i7集显非触屏(NBD-WXX9,NbD-WFH9)原装出厂Win10系统

适用型号&#xff1a;NbD-WFH9、NbD-WFE9A、NbD-WDH9B、NbD-WFE9、 链接&#xff1a;https://pan.baidu.com/s/1qTCbaQQa8xqLR-4Ooe3ytg?pwdvr7t 提取码&#xff1a;vr7t 华为原厂WIN系统自带所有驱动、出厂主题壁纸、系统属性联机支持标志、系统属性专属LOGO标志、Office…

【STIP】安全Transformer推理协议

Secure Transformer Inference Protocol 论文地址&#xff1a;https://arxiv.org/abs/2312.00025 摘要 模型参数和用户数据的安全性对于基于 Transformer 的服务&#xff08;例如 ChatGPT&#xff09;至关重要。虽然最近在安全两方协议方面取得的进步成功地解决了服务 Transf…

leetcode hot100刷题日记——27.对称二叉树

方法一&#xff1a;递归法 class Solution { public:bool check(TreeNode *left,TreeNode *right){//左子树和右子树的节点同时是空的是对称的if(leftnullptr&&rightnullptr){return true;}if(leftnullptr||rightnullptr){return false;}//检查左右子树的值相不相等&a…

高考加油(Python+HTML)

前言 询问DeepSeek根据自己所学到的知识来生成多个可执行的代码&#xff0c;为高考学子加油。最开始生成的都会有点小问题&#xff0c;还是需要自己调试一遍&#xff0c;下面就是完整的代码&#xff0c;当然了最后几天也不会有多少人看&#xff0c;都在专心的备考。 Python励…

贪心算法应用:Ford-Fulkerson最大流问题详解

Java中的贪心算法应用:Ford-Fulkerson最大流问题详解 1. 最大流问题概述 最大流问题(Maximum Flow Problem)是图论中的一个经典问题,旨在找到一个从源节点(source)到汇节点(sink)的最大流量。Ford-Fulkerson方法是解决最大流问题的经典算法之一,它属于贪心算法的范畴…

UE5 Niagara 如何让四元数进行旋转

Axis Angle中&#xff0c;X,Y,Z分别为旋转的轴向&#xff0c;W为旋转的角度&#xff0c;在这里旋转角度不需要除以2&#xff0c;因为里面已经除了&#xff0c;再将计算好的四元数与要进行旋转的四元数进行相乘&#xff0c;结果就是按照原来的角度绕着某一轴向旋转了某一角度

从“黑箱”到透明化:MES如何重构生产执行全流程?

引言 在传统制造企业中&#xff0c;生产执行环节常面临“计划混乱、进度难控、异常频发、数据滞后”的困境。人工派工效率低下、物料错配频发、质量追溯困难等问题&#xff0c;直接导致交付延期、成本攀升、客户流失。深蓝易网MES系统以全流程数字化管理为核心&#xff0c;通过…

探索Linux互斥:线程安全与资源共享

个人主页&#xff1a;chian-ocean 文章专栏-Linux 前言&#xff1a; 互斥是并发编程中避免竞争条件和保护共享资源的核心技术。通过使用锁或信号量等机制&#xff0c;能够确保多线程或多进程环境下对共享资源的安全访问&#xff0c;避免数据不一致、死锁等问题。 竞争条件 竞…

JWT安全:假密钥.【签名随便写实现越权绕过.】

JWT安全&#xff1a;假密钥【签名随便写实现越权绕过.】 JSON Web 令牌 (JWT)是一种在系统之间发送加密签名 JSON 数据的标准化格式。理论上&#xff0c;它们可以包含任何类型的数据&#xff0c;但最常用于在身份验证、会话处理和访问控制机制中发送有关用户的信息(“声明”)。…

Python爬虫实战:抓取百度15天天气预报数据

&#x1f310; 编程基础第一期《9-30》–使用python中的第三方模块requests&#xff0c;和三个内置模块(re、json、pprint)&#xff0c;实现百度地图的近15天天气信息抓取 记得安装 pip install requests&#x1f4d1; 项目介绍 网络爬虫是Python最受欢迎的应用场景之一&…

RV1126 + FFPEG多路码流项目

代码主体思路&#xff1a; 一.VI,VENC,RGA模块初始化 1.先创建一个自定义公共结构体&#xff0c;用于方便管理各个模块 rkmedia_config_public.h //文件名字#ifndef _RV1126_PUBLIC_H #define _RV1126_PUBLIC_H#include <assert.h> #include <fcntl.h> #include …

NodeJS 基于 Koa, 开发一个读取文件,并返回给客户端文件下载,以及读取文件形成列表和文件删除的代码演示

前言 在上一篇文章 《Nodejs 实现 Mysql 数据库的全量备份的代码演示》 中&#xff0c;我们演示了如何将用户的 Mysql 数据库进行备份的代码。但是&#xff0c;这个备份&#xff0c;只是备份在了服务器上了。 而我们用户的真实需求&#xff0c;是需要将备份文件下载到本地进行…

为什么在我的Flask里面有两个路由,但是在网页里有一个却不能正确访问到智能体

1. /zhoushibo 能访问&#xff0c;/chat 直接浏览器访问报 Method Not Allowed 原因&#xff1a; /zhoushibo 路由是你用 app.route(/zhoushibo) 定义的&#xff0c;返回的是一个HTML网页&#xff0c;浏览器访问没问题。 /chat 路由你用的是 app.route(/chat, methods[POST])…

哈工大计算机系统2024大作业——Hello的程序人生

计算机系统 大作业 题 目 程序人生-Hello’s P2P 专 业 人工智能 学   号 2022112040 班 级 2203601 学 生 郄东昕 指 导 教 师 吴锐 计算机科学与技术学院…