跳到主要内容
极客日志极客日志面向AI+效率的开发者社区
首页博客GitHub 精选镜像工具UI配色美学隐私政策关于联系
搜索内容 / 工具 / 仓库 / 镜像...⌘K搜索
注册
博客列表
C++

C++ ODB ORM 核心概念与实战指南

综述由AI生成ODB 作为 C++ 的对象关系映射库,利用编译期代码生成机制实现对象与数据库表的高效映射,规避了运行时反射开销。其支持 MySQL、PostgreSQL 等多种主流数据库,内置事务管理与灵活查询接口。内容涵盖 ODB 环境搭建、核心注解语法、视图定义策略及完整的 CRUD 实战代码示例,旨在协助开发者快速掌握基于 ODB 的数据库持久化方案。

RedisGeek发布于 2026/2/17更新于 2026/6/1217 浏览
C++ ODB ORM 核心概念与实战指南

C++ ODB ORM 核心概念与实战指南

ODB 基本概念

ODB 是一个针对 C++ 的对象关系映射(ORM)库,它允许开发者以面向对象的方式操作数据库,将 C++ 对象与数据库表进行映射,从而避免直接编写 SQL 语句,简化数据库操作。

主要特点:

  • 对象 - 关系映射:将 C++ 类映射到数据库表,类的成员变量映射到表的字段。对象的创建、修改、删除等操作会自动转换为对应的数据库操作(如 INSERT、UPDATE、DELETE)。
  • 代码生成机制:ODB 不依赖运行时反射(C++ 本身不支持),而是通过编译期代码生成实现映射。开发者使用特殊的注解(如 #pragma db object)标记需要持久化的类,然后通过 ODB 编译器生成与数据库交互的代码(如 SQL 语句、CRUD 函数 等)。
  • 支持多数据库:兼容主流数据库,如 MySQL、PostgreSQL、SQLite、Oracle 等。切换数据库时无需修改核心业务代码,只需调整配置。
  • 查询能力:提供类似 SQL 的查询接口(通过 C++ 表达式构建)。例如通过 query 类组合条件,实现复杂查询逻辑,避免手写 SQL。
  • 事务支持:内置事务管理机制,确保数据库操作的原子性、一致性、隔离性和持久性(ACID)。

框架安装

1. 安装 build2

首先安装构建系统 build2。参考官方文档:https://build2.org/install.xhtml

curl -sSfO https://download.build2.org/0.17.0/build2-install-0.17.0.sh
sh build2-install-0.17.0.sh

2. 安装 odb-compiler

注意这里的 gcc 版本需要根据实际环境调整。以下以 gcc-11 为例:

sudo apt-get install gcc-11-plugin-dev
mkdir odb-build && cd odb-build
bpkg create -d odb-gcc-N cc \
    config.cxx=g++ \
    config.cc.coptions=-O3 \
    config.bin.rpath=/usr/lib \
    config.install.root=/usr/ \
    config.install.sudo=sudo
cd odb-gcc-N
bpkg add https://pkg.cppget.org/1/beta
bpkg fetch
bpkg build odb@https://pkg.cppget.org/1/beta
bpkg test odb
bpkg install odb

如果执行 odb --version 报错找不到命令,请添加环境变量:

echo 'export PATH=${PATH}:/usr/local/bin' >> ~/.bashrc
export PATH=${PATH}:/usr/local/bin

验证安装:

odb --version
# 输出示例:ODB object-relational mapping (ORM) compiler for C++ 2.5.0-b.25

3. 安装 ODB 运行时库

cd ..
bpkg create -d libodb-gcc-N cc \
    config.cxx=g++ \
    config.cc.coptions=-O3 \
    config.install.root=/usr/ \
    config.install.sudo=sudo
cd libodb-gcc-N
bpkg add https://pkg.cppget.org/1/beta
bpkg fetch
bpkg build libodb
bpkg build libodb-mysql
bpkg build libodb-boost
bpkg install --all --recursive

常见操作

以下示例以映射到 MySQL 数据库为主。核心头文件如下:

#include <cstddef>      // std::size_t
#include <boost/date_time/posix_time/posix_time.hpp>
#include <odb/nullable.hxx>
#include <odb/core.hxx>

类型映射

通过指令 #pragma 声明 C++ 类与数据库表之间的映射关系,必须放在它们所描述的 C++ 实体(类、数据成员、访问器)之前。格式为:

#pragma db 指令 [参数...]

常用指令及参数说明:

  • object:表示该类将被映射到数据库。
    • table():默认生成的表名就是类名,使用该参数可以指定表名。
  • id:表示该成员将被作为主键。
  • auto:表示该成员有自增属性。
  • unique:表示唯一键索引字段。
  • index:为该字段创建普通索引。
  • null / not_null:控制字段是否允许为空。
  • default():指定默认值。
  • column():指定该成员映射到数据库表后的列名。
  • type():指定该成员映射到数据库表中的字段类型。

示例 person.hxx 文件:

#pragma db object table("person")
class Person {
public:
private:
    friend class odb::access; // ODB 需要访问私有成员,所以需要做友元声明

    #pragma db id auto
    unsigned int _id;

    #pragma db column("user_age") default(18)
    unsigned short _age;

    #pragma db not_null
    std::string _name;

    #pragma db unique type("varchar(20)")
    std::string _phone;
};

编译与建表:

odb -d mysql --generate-query --generate-schema --profile boost/date-time person.hxx

执行后会生成一个 SQL 脚本文件(如 person.sql)。查看内容后,可通过以下命令构建 MySQL 数据库表:

mysql -u root -p 库名 < person.sql

注意事项:

  1. 作用域:#pragma db 只影响紧随其后的类、数据成员或访问器。
  2. 顺序:多个属性可以写在同一行,也可以分开写。
  3. 访问控制:ODB 需要访问私有成员,通常使用 friend class odb::access。
  4. 列名规则:数据库表的列名默认为成员变量名,但会去掉成员名的前缀下划线。

视图映射

ODB 视图是一个只读的、虚拟的投影,它基于一个或多个持久化类(数据库表),通过查询组合出你需要的数据结构。视图本身不对应数据库中的实际表,而是在查询时动态生成的结果集。

核心特点:

  • 只读性:视图只能用于查询,不能进行插入、更新、删除操作。
  • 虚拟性:不占用实际数据库存储空间。
  • 灵活性:可以自由组合多个表的字段,形成新的数据结构。
  • 类型安全:在 C++ 编译期就确定了返回类型。

方式一:基于对象关系的视图

这种视图使用 ODB 的查询语言来定义表之间的关联:

#pragma db view object(Student)\
           object(Classes: Student::classes_id == Classes::id)
struct StudentClassView {
    #pragma db column(Student::name) std::string student_name;
    #pragma db column(Student::age) unsigned short student_age;
    #pragma db column(Classes::name) std::string class_name;
};

工作方式:

  • object(Student):指定主表。
  • object(Classes: Student::classes_id == Classes::id):指定关联表及连接条件。

相当于 SQL:

SELECT Student.name, Student.age, Classes.name 
FROM Student JOIN Classes ON Student.classes_id = Classes.id

方式二:基于原生 SQL 的视图

当需要复杂查询时,可以直接使用原生 SQL:

#pragma db view query("SELECT s.name, s.age, c.name "\
                      "FROM student s "\
                      "JOIN classes c ON s.classes_id = c.id "\
                      "WHERE s.age > ?")
struct AdultStudentView {
    std::string student_name;
    unsigned short student_age;
    std::string class_name;
};

特点:

  • 使用 query("原生 SQL") 定义。
  • ? 是参数占位符,可以在查询时传入具体值。
  • 适合复杂的、对象关系无法表达的查询。

ODB 类与接口

  • odb::database 类:实现数据库的增删改查操作。包含构造函数、事务管理、查询操作、单对象操作等成员函数。
  • odb::result 类:处理查询结果集。

测试示例

数据库操作流程一般遵循以下步骤:

  1. 构造连接池对象。
  2. 构造数据操作 database 对象。
  3. 获取事务对象,开启事务。
  4. 执行数据库操作。
  5. 提交事务(若无显式添加,在销毁前会进行事务回滚)。

头文件定义 (Student.hpp)

#pragma once
#include <string>
#include <cstddef>
#include <boost/date_time/posix_time/posix_time.hpp>
#include <odb/nullable.hxx>
#include <odb/core.hxx>

class Student {
private:
    unsigned long _id;
    std::string _sn;
    std::string _name;
    unsigned short _age;
    unsigned long _classes_id;
};

class Classes {
private:
    unsigned long _id;
    std::string _name;
};

#pragma db view object(Student)\
           object(Classes classes: Student._classes_id == Classes._id)\
           query((?))
struct classes_student {
    #pragma db column(Student::_id) unsigned long _id;
    #pragma db column(Student::_sn) std::string _sn;
    #pragma db column(Student::_name) std::string _name;
    #pragma db column(Student::_age) unsigned short _age;
    #pragma db column(Classes::_name) std::string _classes_name;
};

#pragma db view query("select name from Student " + (?))
struct all_name {
    std::string _name;
};

测试代码 (test.cc)

#include <gflags/gflags.h>
#include <odb/database.hxx>
#include <odb/mysql/database.hxx>
#include "student_classes.hxx"
#include "student_classes-odb.hxx"

DEFINE_string(host, "127.0.0.1", "主机号");
DEFINE_uint32(port, 0, "端口号");
DEFINE_string(db, "qsy_test", "mysql 数据库名");
DEFINE_string(user, "root", "mysql 用户名");
DEFINE_string(pswd, "your_password", "mysql 用户密码");
DEFINE_string(cset, "utf8", "mysql 客户端字符集");
DEFINE_int32(max_pool, 3, "最大的连接池数");

void insert_student(odb::mysql::database &db) {
    try {
        odb::transaction trans(db.begin());
        Student s1(1, "张三", 18, 1);
        Student s2(2, "李四", 19, 1);
        Student s3(3, "王五", 17, 1);
        Student s4(4, "赵六", 21, 2);
        Student s5(5, "田七", 20, 2);
        Student s6(6, "孙八", 16, 2);
        Student s7(7, "罗九", 26, 2);
        db.persist(s1);
        db.persist(s2);
        db.persist(s3);
        db.persist(s4);
        db.persist(s5);
        db.persist(s6);
        db.persist(s7);
        trans.commit();
    } catch (std::exception &e) {
        std::cout << "数据插入失败:" << e.what() << std::endl;
    }
}

void insert_classes(odb::mysql::database& db) {
    try {
        odb::transaction trans(db.begin());
        Classes c1("1 班");
        Classes c2("2 班");
        db.persist(c1);
        db.persist(c2);
        trans.commit();
    } catch (const std::exception& e) {
        std::cout << "数据插入失败" << e.what() << '\n';
    }
}

void remove_student(odb::mysql::database &db) {
    try {
        odb::transaction trans(db.begin());
        typedef odb::query<Student> query;
        db.erase_query<Student>(query::id == 4);
        trans.commit();
    } catch (const std::exception &e) {
        std::cerr << "删除失败:" << e.what() << '\n';
    }
}

Student select_student(odb::mysql::database &db) {
    Student ret;
    try {
        odb::transaction trans(db.begin());
        typedef odb::query<Student> query;
        typedef odb::result<Student> result;
        result r(db.query<Student>(query::name == "张三"));
        if (r.size() != 1) {
            std::cout << "查询失败" << std::endl;
            return Student();
        }
        ret = *r.begin();
        trans.commit();
    } catch (const std::exception &e) {
        std::cerr << "查询失败" << e.what() << '\n';
    }
    return ret;
}

void select_student_classes(odb::mysql::database& db) {
    try {
        odb::transaction trans(db.begin());
        typedef odb::query<struct classes_student> query;
        typedef odb::result<struct classes_student> result;
        result r(db.query<struct classes_student>(query::classes::id == 1));
        for (auto it = r.begin(); it != r.end(); it++) {
            std::cout << it->name << " ";
            std::cout << it->age << " ";
            std::cout << it->sn << " ";
            std::cout << it->classes_name << " ";
            std::cout << std::endl;
        }
        trans.commit();
    } catch (const std::exception &e) {
        std::cout << "查询失败:" << e.what() << std::endl;
    }
}

void update_student(odb::mysql::database &db, Student &stu) {
    try {
        odb::transaction trans(db.begin());
        db.update(stu);
        trans.commit();
    } catch (const std::exception &e) {
        std::cerr << "更新失败:" << e.what() << '\n';
    }
}

int main(int argc, char* argv[]) {
    google::ParseCommandLineFlags(&argc, &argv, true);
    // 创建连接池构建配置对象
    std::unique_ptr<odb::mysql::connection_pool_factory> cpf(
        new odb::mysql::connection_pool_factory(FLAGS_max_pool, 0));
    // 创建数据操作对象
    odb::mysql::database db(FLAGS_user, FLAGS_pswd, FLAGS_db, FLAGS_host, FLAGS_port, "", FLAGS_cset, 0, std::move(cpf));

    // 插入
    // insert_student(db);
    // insert_classes(db);
    // 更新
    // Student stu = select_student(db);
    // stu.age(5);
    // update_student(db, stu);
    // 删除
    // remove_student(db);
    // 查询
    select_student_classes(db);
    return 0;
}

Makefile

test : test.cc student_classes-odb.cxx
	c++ -g $^ -o $@ -lodb-mysql -lodb -lodb-boost -lgflags

以上就是 ODB 的核心用法与实战示例。在实际项目中,建议根据业务需求合理设计持久化类,并注意事务边界的管理,以确保数据的一致性。

目录

  1. C++ ODB ORM 核心概念与实战指南
  2. ODB 基本概念
  3. 框架安装
  4. 1. 安装 build2
  5. 2. 安装 odb-compiler
  6. 输出示例:ODB object-relational mapping (ORM) compiler for C++ 2.5.0-b.25
  7. 3. 安装 ODB 运行时库
  8. 常见操作
  9. 类型映射
  10. 视图映射
  11. ODB 类与接口
  12. 测试示例
  13. 头文件定义 (Student.hpp)
  14. 测试代码 (test.cc)
  15. Makefile
  • 💰 8折买阿里云服务器限时8折了解详情
  • Magick API 一键接入全球大模型注册送1000万token查看
  • 🤖 一键搭建Deepseek满血版了解详情
  • 一键打造专属AI 智能体了解详情
极客日志微信公众号二维码

微信扫一扫,关注极客日志

微信公众号「极客日志V2」,在微信中扫描左侧二维码关注。展示文案:极客日志V2 zeeklog

更多推荐文章

查看全部
  • CSS 绘制圆形与三角形技巧:border 与 border-radius 实战
  • LFM2.5-1.2B-Thinking 模型效果展示与性能分析
  • Windows11 使用 llama.cpp 部署 Qwen3.5 量化模型测试
  • 前端数据库 IndexedDB 详解:构建离线 Web 应用
  • 8 款 AI 写作工具小说创作能力横评
  • Llama.cpp 全实战指南:跨平台部署本地大模型的零门槛方案
  • VR、具身智能与人形机器人:构建现实世界的智能接口
  • 算法空间复杂度详解:概念与常见计算实例
  • Chatbot UI 实战指南:基于 Web Components 构建高可用对话界面
  • C++ 网络编程中的 Protobuf 消息分发 (Dispatcher) 设计模式
  • ICCV2019 论文解读:无需数据集的 Student Networks 训练方法
  • Java+Leaflet 实现湖南省道路长度 WebGIS 系统
  • WebRTC 技术详解:架构、组件与网络穿透
  • PHP 对接 DeepSeek API 实现指南
  • FANUC 机器人 PR 寄存器详解
  • Cookie 机制深度解析:原理、版本与局限
  • 颜色透明度百分比与十六进制换算指南
  • Python 自学指南:从零基础到数据分析与人工智能实战路径
  • HTTP 协议核心:请求方法与 Cookie 状态管理
  • 网页开发常用知识点梳理

相关免费在线工具

  • Base64 字符串编码/解码

    将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online

  • Base64 文件转换器

    将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online

  • Markdown转HTML

    将 Markdown(GFM)转为 HTML 片段,浏览器内 marked 解析;与 HTML转Markdown 互为补充。 在线工具,Markdown转HTML在线工具,online

  • HTML转Markdown

    将 HTML 片段转为 GitHub Flavored Markdown,支持标题、列表、链接、代码块与表格等;浏览器内处理,可链接预填。 在线工具,HTML转Markdown在线工具,online

  • JSON 压缩

    通过删除不必要的空白来缩小和压缩JSON。 在线工具,JSON 压缩在线工具,online

  • JSON美化和格式化

    将JSON字符串修饰为友好的可读格式。 在线工具,JSON美化和格式化在线工具,online