TinyORM:一个简单的C++ ORM框架

项目地址:https://github.com/david-pp/tiny-orm

TinyORM

对象关系映射(ORM,Object-Relational Mappings),是一种程序技术,用于实现面向对象编程语言里不同类型系统的数据之间的转换。说白了,就是将C++对象映射到数据库表的一种技术,目的就是简化对象存档和交换的代码。

一般情况下,对象要存储到关系型数据库,需要在代码里面使用SQL语句,若对象较为复杂的话,就需要编写大量代码去支持对象的数据库操作,代码也不怎么美观,扩展性也比较差。ORM则封装这些操作作为对象的成员函数,这样使用时候就会比较方便。(PS. ORM会损失一定的灵活性,偶尔SQL也是必要的)

该项目尝试使用C++写一个简单的ORM,支持:

  • 一个C++结构体映射射到一张关系表。
  • 映射关系可以手工编写也可以使用工具自动生成。
  • 结构体成员变量支持多种类型,简单类型、复合类型、STL容器都有支持。
  • 复合成员或STL成员映射为表的二进制字段,此时需进行序列化,当然序列化也可以自动生成。
  • 数据库表的自动创建和更新(现阶段只支持数据库添加字段)。

不支持复杂的多表操作,数据分表等,用于存档的话完全够用了。

依赖:

  • C++11
  • protobuf
  • mysql/mysql++soci

框架和简单用法:

tinyorm.png

用法

现在要定义一个游戏玩家对象,有名字、ID、年龄,同时可以装备多件武器。

第一步:定义支持映射的结构体

// 武器数据
struct Weapon {
    uint32_t type = 0;
    std::string name = "";

    std::string serialize() const {
        WeaponProto proto;
        proto.set_type(type);
        proto.set_name(name);
        return proto.SerializeAsString();
    }

    bool deserialize(const std::string &data) {
        WeaponProto proto;
        if (proto.ParseFromString(data)) {
            type = proto.type();
            name = proto.name();
            return true;
        }
        return false;
    }
};

// 玩家数据
struct Player {
    uint32_t id = 0;
    std::string name;
    uint8_t age = 0;
    std::vector<Weapon> weapons;
}

第二步:定义映射关系

Player结构体映射到PLAYER表:

> DESC PLAYER;
+----------------+----------------------+------+-----+---------+-------+
| Field          | Type                 | Null | Key | Default | Extra |
+----------------+----------------------+------+-----+---------+-------+
| ID             | int(10) unsigned     | NO   | PRI | 0       |       |
| NAME           | text                 | YES  |     | NULL    |       |
| AGE            | tinyint(3)           | NO   |     | 30      |       |
| WEAPONS        | blob                 | YES  |     | NULL    |       |
+----------------+----------------------+------+-----+---------+-------+
TableFactory::instance().table<Player>("PLAYER")
        .field(&Player::id, "ID", FieldType::UINT32)
        .field(&Player::name, "NAME", FieldType::STRING)
        .field(&Player::age, "AGE", FieldType::UINT8, "30")
        .field(&Player::weapons, "WEAPONS", FieldType::OBJECT)
        .key("ID")
        .index("NAME");

第三步:直接使用

假设要把所有PLAYER加载到内存:

TinyORM db;
TinyORM::Records<Player> players;
db.loadFromDB(players);

假设要更新PLAYER表里面的某条记录:

Object2DB<Player> p;
p.id = 1024;
p.age = 30;
p.name = "David";
p.weapons.clear();
p.update();

例子详细内容,请查看:demo_orm.cpp

自动生成ORM映射关系和序列化代码

第一步:设计对象并填写用于生成ORM的配置

<?xml version="1.0" encoding="UTF-8"?>
<tinyobj>
    <!--<import>tinyplayer.xml</import>-->

    <Weapon orm="0" keys="id" index="" comment="武器">
        <type    num="1" type="uint32"  comment="标识符"/>
        <name    num="2" type="string"  comment="名字"/>
    </Weapon>

    <Player keys="id" index="name" comment="角色信息">
        <id      num="1" type="uint32"  comment="标识符"/>
        <name    num="2" type="string"  default="david" comment="名字"/>
        <age num="3" type="uint8" comment="国家信息"/>
        <weapons num="4" type="std::vector{Weapon}" comment="武器列表"/>
    </Player>
</tinyobj>

第二步:生成ORM和序列化代码

tools/tinyobj.py -o example/tinyobj/ example/tinyobj/tinyplayer.xml

将会在example/tinyobj目录下生成下列文件:

  • 对象定义文件:tinyplayer.h tinyplayer.cpp
  • 对象序列化相关文件:tinyplayer.proto tinyplayer.pb.h tinyplayer.pb.cc
  • ORM映射:tinyplayer.orm.h

例子详情,请查看:tinyplayer.xml

自动生成配置文件说明

字段类型支持(type字段)

  • 基本类型:
    脚本类型 : (C++类型、PROTO类型、DB字段类型、默认值)

    # 整数-有符号
    "int8" : ("int8_t",  "sint32", "INT8", "0"),
    "int16": ("int16_t", "sint32", "INT16", "0"),
    "int32": ("int32_t", "sint32", "INT32", "0"),
    "int64": ("int64_t", "sint64", "INT64", "0"),

    # 整数-无符号
    "uint8" : ("uint8_t",  "uint32", "UINT8", "0"),
    "uint16": ("uint16_t", "uint32", "UINT16", "0"),
    "uint32": ("uint32_t", "uint32", "UINT32", "0"),
    "uint64": ("uint64_t", "uint64", "UINT64", "0"),

    # 浮点
    "float" : ("float", "float", "FLOAT", "0"),
    "double": ("double", "double", "DOUBLE", "0"),

    # 布尔
    "bool": ("bool", "bool", "BOOL", "0"),

    # 字符串
    "string": ("std::string", "bytes", "STRING", ""),
    "vchar" : ("std::string", "bytes", "VCHAR", ""),

    # 二进制
    "bytes"         : ("std::string", "bytes", "BYTES", ""),
    "bytes_tiny"    : ("std::string", "bytes", "BYTES_TINY", ""),
    "bytes_medium"  : ("std::string", "bytes", "BYTES_MEDIUM", ""),
    "bytes_long"    : ("std::string", "bytes", "BYTES_LONG",""),
  • 自定义类型:
Player
Weapon
  • STL:(由于<>两个字符在XML中是特殊字符,脚本中表示STL容器时用{}替换即可)
// 简单
std::vector{int8}
// 复合类型
std::list<Player>
// 任意层次的嵌套
std::map{std::map{std::string, Player}}

指定表的键值和索引(现阶段尚未支持索引)

<Player keys="id" index="name" comment="角色信息">
    ...
</Player>

控制不需要生成ORM映射

orm="0"

<Weapon orm="0" keys="id" index="" comment="武器">
    ...
</Weapon>

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容