ODB 是一个针对 C++ 的对象关系映射(ORM)库,它允许开发者以面向对象的方式操作数据库,将C++ 对象与数据库表进行映射,从而避免直接编写 SQL 语句,简化数据库操作。
特点:
INSERT、UPDATE、DELETE)。#pragma db object)标记需要持久化的类,然后通过 ODB 编译器生成与数据库交互的代码(如 SQL 语句、CRUD 函数等)。MySQL、PostgreSQL、SQLite、Oracle 等,切换数据库时无需修改核心业务代码,只需调整配置。query 类组合条件,实现复杂查询逻辑,避免手写 SQL。build2安装: 安装步骤: https://build2.org/install.xhtml#unix
dev@dev-host:~/workspace curl -sSfO https://download.build2.org/0.17.0/build2-install-0.17.0.sh dev@dev-host:~/workspace sh build2-install-0.17.0.sh
安装 odb-compiler:
dev@dev-host:~/workspace #注意这里的 gcc-11 需要根据你自己版本而定 dev@dev-host:~/workspace sudo apt-get install gcc-11-plugin-dev dev@dev-host:~/workspace mkdir odb-build && cd odb-build dev@dev-host:~/workspace/odb-build cd odb-gcc-N dev@dev-host:~/workspace/odb-build/odb-gcc-N bpkg build odb@https://pkg.cppget.org/1/beta dev@dev-host:~/workspace/odb-build/odb-gcc-N bpkg test odb test odb-2.5.0-b.25+1/tests/testscript{testscript} tested odb/2.5.0-b.25+1 dev@dev-host:~/workspace/odb-build/odb-gcc-N bpkg install odb dev@dev-host:~/workspace/odb-build/odb-gcc-N odb --version bash: /usr/bin/odb: No such file or directory #如果报错了,找不到 odb,那就在执行下边的命令 dev@dev-host:~/workspace/odb-build/odb-gcc-N sudo echo 'export PATH={PATH}:/usr/local/bin' >> ~/.bashrc dev@dev-host:~/workspace/odb-build/odb-gcc-N{PATH}:/usr/local/bin dev@dev-host:~/workspace/odb-build/odb-gcc-N odb --version ODB object-relational mapping (ORM) compiler for C++ 2.5.0-b.25 Copyright (c) 2009-2023 Code Synthesis Tools CC. This is free software; see the source for copying conditions. There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
安装 ODB 运行时库:
dev@dev-host:~/workspace/odb-build/odb-gcc-N cd .. dev@dev-host:~/workspace/odb-build bpkg create -d libodb-gcc-N cc \ config.cxx=g++ \ config.cc.coptions=-O3 \ config.install.root=/usr/ \ config.install.sudo=sudo dev@dev-host:~/workspace/odb-build cd libodb-gcc-N dev@dev-host:~/workspace/odb-build/libodb-gcc-N bpkg add https://pkg.cppget.org/1/beta dev@dev-host:~/workspace/odb-build/libodb-gcc-N bpkg fetch dev@dev-host:~/workspace/odb-build/libodb-gcc-N bpkg build libodb dev@dev-host:~/workspace/odb-build/libodb-gcc-N
安装 boost profile 库:
dev@dev-host:~/workspace/odb-build/libodb-gcc-N$
bpkg build libodb-boost
总体打包安装:
dev@dev-host:~/workspace/odb-build/libodb-gcc-N$
bpkg install --all --recursive
提示:接下以映射到mysql数据库为示例演示 核心头文件
/*
在 C++ 中,要使用 ODB 将类声明为持久化类,需要包含 ODB 的核心头文
件,并使用 #pragma db object 指令
#pragma db object 指示 ODB 编译器将 person 类视为一个持久化类。
*/
#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脚本文件,通过指令cat person.sql查看该文件

通过该文件构建mysql数据库表:
mysql -u root -p 库名 < person.sql
登陆mysql数据库查看表属性:

注意1:
#pragma db 只影响紧随其后的类、数据成员或访问器friend class odb::access注意2:
视图映射 ODB视图是一个只读的、虚拟的投影,它基于一个或多个持久化类(数据库表),通过查询组合出你需要的数据结构。视图本身不对应数据库中的实际表,而是在查询时动态生成的结果集。 视图的核心特点
视图的两种定义方式 方式一:基于对象关系的视图 这种视图使用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::database类:实现数据库的增删改查操作
构造函数
// MySQL
odb::mysql::database(
const std::string& user,
const std::string& password,
const std::string& database,
const std::string& host = "",
unsigned int port = 0
);成员函数
事务管理
odb::transaction begin();//获取并开启事务
//odb::transaction类
void commit();//事务提交
void rollback();//事务回滚查询操作
result<T> query(const odb::query<T>&);
result<T> query(const odb::query<T>&, const odb::query_params&);单对象操作
void persist(T& object); // 插入
void update(const T& object); // 更新
void erase(const T& object); // 删除odb::result 类 :查询结果集
// 迭代器
iterator begin();
iterator end();
// 容量查询
bool empty() const;
std::size_t size() const;
// 遍历操作
T& front(); // 获取第一个对象
void next(); // 移动到下一个对象
bool done() const; // 检查是否遍历完成数据库操作流程:
Student.hpp
#pragma once
#include <string>
#include <cstddef> // std::size_t
#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, "I5sLBKpqjGjPZi", "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 updata_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);
// updata_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非常感谢您能耐心读完这篇文章。倘若您从中有所收获,还望多多支持呀!
我的博客即将同步至腾讯云开发者社区,邀请大家一同入驻:https://cloud.tencent.com/developer/support-plan?invite_code=45sbqgxtzjv