Python的魔法ORM --《PonyORM教程》 2 实体关系

关系的声明和映射

实体之间的关系映射其实就是数据库的外键关联。一般分为:

  • 一对一
  • 一对多
  • 多对多

三大类。

题外话:外键

在码农的队伍里,有相当一部分外键过敏人士。他们觉得外键设置麻烦,使用和删除更麻烦,特别是那种嵌套几层的外键关联设计,添加修改删除上的各种约束,🖕🖕🖕🖕。不过吐槽归吐槽,自打外键诞生以来,这么多程序员,几十年的发展。如果外键没什么用处,也不会成为sql数据库的标配了,甚至很多nosql的数据库,都有类似外键的对象的存在。其实外键在数据一致性,查询速度等方面带来的优势是巨大的。甚至是程序员用编程的方法很难弥补的。哪怕你是全球反外键联盟的资深人士,我还是建议你,出于节省时间和简化逻辑起见,尽量利用外键的设计给自己的开发工作带来便利(or 噩梦! 😝)

pony的关系映射尊从简单直接的原则,外键中的id字段,会被映射成实体关系。下面上代码,给大家实例说明:

假设实体之间的关系如下:

  1. Father(父亲)和Mother(母亲)是一对一的关系: 一夫一妻制
  2. Father(父亲),Mother(母亲)和Child(孩子)之间是一对多的关系: 一对夫妻可以生多个孩子
  3. Child(孩子)和Uncle(叔叔)之间是多对多的关系: 一个孩子可以有多个叔叔,一个叔叔也可能有多个侄子
  4. Child(孩子)和Child(孩子)之间是多对多的关系。每个孩子都可以有多个朋友

开始上代码,我们先建立四个基本模型:Father, Mother, Child和Uncle。

class Father(db.Entity):
    _table_ = "father"
    name = Required(str, max_len=40)


class Mother(db.Entity):
    _table_ = "mother"
    name = Required(str, max_len=40)


class Child(db.Entity):
    _table_ = "child"
    name = Required(str, max_len=40)


class Uncle(db.Entity):
    _table_ = "uncle"
    name = Required(str, max_len=40)

我们先在Father, Mother之间声明一对一的关系。这需要在2个类中各增加一个字段用于记录外键并映射关系。修改后的代码如下:

class Father(db.Entity):
    _table_ = "father"
    name = Required(str, max_len=40)
    wife = Optional("Mother", nullable=True, default=None, column="wife_id", reverse="husband")  # 老婆, 一对一


class Mother(db.Entity):
    _table_ = "mother"
    name = Required(str, max_len=40)
    husband = Optional("Father", nullable=True, default=None, column="husband_id", reverse="wife")  # 老公, 一对一

wife和husband是关系映射。我们从Father.wife字段举例进行说明:

  • wife 映射关系名称,我们可以直接从一个Father的wife字段访问对应的Mother实例。
  • Optional 表明在创建Father实例时wife是一个可选字段,
  • Mother 指明了这个映射的对象是 class Mother,由于class Mother定义在class Father下面,所以这里用了引号,避免引用错误。
  • nullable 就是数据库字段定义中的 not null,一般如果时个Optional字段,这里做好都定义为 nullable=True
  • default 默认值
  • column 数据库的表中,此列的名字
  • reverse 反向引用。对于ORM不了解的同学,可以对这个概念比较陌生,这个参数只在有映射关系的时候才会定义。意思是 本对象关系映射的另一方的字段名,通俗的举个例子
  • 小明(Father的实例)和小红(Mother的实例)是两口
  • 你如果从小明的角度去看小红,那么小红就是 小明的老婆(wife),反过来小明就是小红的husband, 这个reverse就是从对方的角度来看和自己建立映射的字段的名字。

如果两个实体的映射关系只有一对,不用设置这个reverse系统本身也能推断出来对应的字段。但是,如果2个实体之间有多个映射关系,那就必须定义reverse了。个人的经验是:无论何时都定义reverse

接下来我们分别创建Father和Mother的实例,并给他们建立映射关系

    with db_session:
        fa = Father(name="小明")
        mo = Mother(name="小红")
        mo.flush()  # 生成id
        fa.wife = mo
        commit()  # 提交,通常这不是必须的这里只是为了演示
        print(Father.get().wife.to_dict())
        print(Mother.get().husband.to_dict())
{'id': 3, 'name': '小红', 'husband': 3}
{'id': 3, 'name': '小明', 'wife': 3}
  • flush 实例创建后,由于没有写入数据库,所以仅仅时保留 在上下文中,并没有生成id,mo.flush() 方法用于把当前实例刷入数据库并获取一个id。事实上,你不用考虑何时执行这个flush()动作,因为只要你一个rollback,所有之前的操作都会cancel,rollback是pony的亮点之一,至于为何闪亮,就留着读者在日后的工作中慢慢体会吧。
  • 赋值 fa.wife = mo 这实际上就是一个赋值操作,就是这个操作,完成了实体映射双方的字段的修改。记住,把一个实体的实例赋值给另一个实例的前提条件是右边的实例必须有id,所以上面 mo.flush() 这一步的操作是必须的。

Father.get().wife.to_dict() 的意思就是从Father中查找一个对象并把他的wife属性转换成字典打印出来。为什么要转换成字典呢?因为这个wife是一个Mother对象!👌,自己动手试一试,就知道pony在背后做了多少工作。

ok,就是这么简单,上面就是基本的1对1的关系的操作。

接下来,演示一下一对多的关系,修改Father,Mother和Child:

class Father(db.Entity):
    _table_ = "father"
    name = Required(str, max_len=40)
    wife = Optional("Mother", nullable=True, default=None, column="wife_id", reverse="husband")  # 老婆, 一对一
    children = Set("Child", reverse="father")  # 孩子, 一对多


class Mother(db.Entity):
    _table_ = "mother"
    name = Required(str, max_len=40)
    husband = Optional("Father", nullable=True, default=None, column="husband_id", reverse="wife")  # 老公, 一对一
    children = Set("Child", reverse="mother")  # 孩子, 一对多


class Child(db.Entity):
    _table_ = "child"
    name = Required(str, max_len=40)
    father = Optional("Father", nullable=True, default=None, column="father_id", reverse="children")  # 父亲 多对一
    mother = Optional("Mother", nullable=True, default=None, column="mother_id", reverse="children")  # 母亲 多对一

Father和Mother都新增了一个children的属性的定义,请注意这个属性的定义的类型是Set,这代表着children属性是一个集合,并且拥有去重的特性。
Child新增了father, mother两个属性,请注意Optional和Required都会在数据库的表中生成对应的字段。使用 column 参数来定义这个字段名,如果你不定义这个column参数,pony会直接使用属性名作默认值。

下面,我们创建一个Father实例,一个Mother实例和两个Child,然后在他们之间建立关系。

if __name__ == "__main__":
    db.drop_table(table_name="father", if_exists=True, with_all_data=True)  # 删除表,演示实体声明时用于快速清除旧表
    db.drop_table(table_name="mother", if_exists=True, with_all_data=True)  # 删除表,演示实体声明时用于快速清除旧表
    db.drop_table(table_name="child", if_exists=True, with_all_data=True)  # 删除表,演示实体声明时用于快速清除旧表
    db.generate_mapping(create_tables=True)  # 生成实体,表和映射关系
    with db_session:
        father = Father(name="李雷")
        mother = Mother(name="韩梅梅")
        child1 = Child(name="乔治")
        child2 = Child(name="佩奇")
        flush()  # 刷新,这里的刷新是为了让刚才创建的对象分配到id
        father.wife = mother  # 李雷的老婆是韩梅梅
        child1.father = father  # 乔治的爸爸是李雷
        child1.mother = mother  # 乔治的妈妈是韩梅梅
        child2.father = father  # 佩奇的爸爸是李雷
        child2.mother = mother  # 乔治的妈妈是韩梅梅
        fa = Father.get(name="李雷")  # 查询叫李雷的Father(父亲)实例
        ch = Child.get(name="乔治")  # 查询叫乔治的Child(孩子)实例
        print(fa.name + "的孩子是" + "和".join([x.name for x in fa.children]))
        print(ch.name + "的父母是{}和{}".format(ch.father.name, ch.mother.name))
    pass

运行结果


Connected to pydev debugger (build 192.6603.34)
李雷的孩子是佩奇和乔治
乔治的父母是李雷和韩梅梅

Process finished with exit code 0

上面的代码,我们创建了一对父母(李雷和韩梅梅)和2个孩子(佩奇和乔治),然后建立他们之间的关系的映射。


屏幕截图4.png

接下来,我们演示的是多对多的关系映射。在这里,我们新引进了一个实体:Uncle(叔叔),和Child实体之间有多对多的关系
一个孩子可以有多个叔叔,一个叔叔也可以有多个侄子

class Child(db.Entity):
    _table_ = "child"
    name = Required(str, max_len=40)
    father = Optional("Father", nullable=True, default=None, column="father_id", reverse="children")  # 父亲 多对一
    mother = Optional("Mother", nullable=True, default=None, column="mother_id", reverse="children")  # 母亲 多对一
    uncles = Set("Uncle", reverse="nephews", table="nephews_uncles")  # 叔叔, 多对多


class Uncle(db.Entity):
    _table_ = "uncle"
    name = Required(str, max_len=40)
    nephews = Set("Child", reverse="uncles", table="nephews_uncles")  # 侄子, 多对多

你可能注意到了,在Set的定义中多了一个table参数,这是由于在多对多的关系中,需要第三张表辅助完成这种关系的映射,这里定义的是第三张表的名称,当然,这个参数也有默认值,那就是A表的表明+_+B表的表名。在本例中,默认的第三张表的表名是child_uncle

接下来,我们分别创建2个Uncle和Child对象,然后把他们建立多对多的关系

if __name__ == "__main__":
    db.drop_table(table_name="uncle", if_exists=True, with_all_data=True)  # 删除表,演示实体声明时用于快速清除旧表
    db.drop_table(table_name="child", if_exists=True, with_all_data=True)  # 删除表,演示实体声明时用于快速清除旧表
    db.generate_mapping(create_tables=True)  # 生成实体,表和映射关系
    with db_session:
        uncle1 = Uncle(name="约翰")  # 创建一个Uncle对象的实例
        uncle2 = Uncle(name="哈特")  # 创建一个Uncle对象的实例
        child1 = Child(name="乔治")  # 创建一个Child对象的实例
        child2 = Child(name="佩奇")  # 创建一个Child对象的实例
        flush()  # 刷新,这里的刷新是为了让刚才创建的对象分配到id
        uncle1.nephews = [child1, child2]
        uncle2.nephews = [child1, child2]
        un = Uncle.get(name="约翰")  # 查询叫约翰的Uncle(叔叔)实例
        ch = Child.get(name="乔治")  # 查询叫乔治的Child(孩子)实例
        print(un.name + "的侄子是" + "和".join([x.name for x in un.nephews]))
        print(ch.name + "的叔叔是" + "和".join([x.name for x in ch.uncles]))
    pass

执行结果

Connected to pydev debugger (build 192.6603.34)
约翰的侄子是佩奇和乔治
乔治的叔叔是约翰和哈特

Process finished with exit code 0

查询数据库,你会发现pony自动生成了第三张多对多的表,并且在其中插入了数据。

最后介绍一种关系自引用
我们假设每个人(Person)都有多个朋友。(人有朋友很正常,😄)

class Person(db.Entity):
    _table_ = "person"
    name = Required(str, max_len=40)
    lover = Optional("Person", default=None, nullable=True, reverse="lover")  # 情人, 自引用,一对一
    friends = Set("Person", reverse="friends", table="friend")  # 朋友, 自引用,多对多


if __name__ == "__main__":
    db.drop_table(table_name="person", if_exists=True, with_all_data=True)  # 删除表,演示实体声明时用于快速清除旧表
    db.generate_mapping(create_tables=True)  # 生成实体,表和映射关系
    with db_session:
        person1 = Person(name="约翰")  # 创建一个Person对象的实例
        person2 = Person(name="玛丽")  # 创建一个Person对象的实例
        person3 = Person(name="乔治")  # 创建一个Person对象的实例
        flush()  # 刷新,这里的刷新是为了让刚才创建的对象分配到id
        person1.lover = person2  # 约翰和玛丽是情人关系
        person3.friends = [person1, person2]  # 约翰和玛丽都是乔治的朋友
        p1 = Person.get(name="约翰")  # 查询叫约翰的Person实例
        p2 = Person.get(name="乔治")  # 查询叫乔治的Person实例
        print("{}和{}都是{}的朋友".format(p1.name, p1.lover.name, " ".join([x.name for x in p1.friends])))
        print(p2.name + "的朋友是" + "和".join([x.name for x in p2.friends]))
    pass
Connected to pydev debugger (build 192.6603.34)
约翰和玛丽都是乔治的朋友
乔治的朋友是约翰和玛丽

Process finished with exit code 0

在Person的定义中,大家可以看到,lover和friends都是自引用对象,只不过一个是一对一,一个是多对多的关系。同样,多对多需要第三张表辅助完成(使用table参数定义),但这个参数你可以不定义。pony会提供默认值(在本例中,这个默认值是person_friends)

索引

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

推荐阅读更多精彩内容