郭英义有些疑惑,他的亲兵不是不知道他的习惯,还这样坚持的话,或许真有什么重要事情。

“孙头,不好了,有人从楼上掉下来了。”王小民一边说着,一边往屋里瞅。

专访诺心创始人张岚:安心美味的蛋糕故事

“何止严重,就算你要逆转轮回帮一个十世善人,属于多福多寿,福禄无双的人逆转一次轮回所承受的代价都大得吓死人,更别说是贞子之前犯下的罪孽那么吓人,帮她逆转的话除非是变成僵尸,不然的话让她以正常人的方式重生的话比起帮其他人代价更大,承受的反噬更恐怖。
他的手臂一直搭在邪月的肩膀上,鬼斗罗命令一下,唐三搭在邪月肩头的手臂骤然收紧,食指化为玉色,直接点在了邪月胸口上。

秦圣嘴角微微抽了抽说道:“老板,李文静可是我们集团的重要员工,你可不要既伤了人家的人,又伤了人家的心。”

Michael


什么是ORM?

ORM的英文全称是“Object Relational Mapping”,即对象-关系映射,从字面上直接理解,就是把“关系”给“对象”化

对应到数据库,我们知道关系数据库(例如Mysql)的特征就是数据与数据之间存在各种各样的“关系”,这种“关系”是由Table(表)来维护和表现的。

ORM就是把关系数据库的一个"表"映射成一个"类",然后给"类"添加各种各样的方法(比如增删改查)

这样,写代码更简单,不用直接操作SQL语句。

要编写一个ORM框架,所有的类都只能动态定义,因为我们知道,在数据库使用过程中,Mysql的“表”是由User来创建的;对应过来,那么“表”对应的类也是应该由User来创建的。

让我们来尝试编写一个ORM框架。

编写底层模块的第一步,就是先把调用接口写出来。

先搞清楚user会如何使用这个ORM框架:使用这个ORM框架,创建一个Customer类来操作对应的数据库表customer,我们期待他写出这样的代码:

class Customer(Model):
    # 定义类的属性到列的映射:
    id = IntegerField("id")
    name = StringField("username")
    email = StringField("email")
    password = StringField("password")

# 创建一个实例:
u = Customer(id=12345, name="Michael", email="test@orm.org", password="my-pwd")
# 保存到数据库:
u.save()

 

其中,父类Model和属性类型StringFieldIntegerField是由ORM框架提供的。显然,save()等方法应该由ORM提供,这样既安全又使得user不用重复定义这些方法。具体实现上,save()等全部由metaclass和基类Model自动完成。虽然metaclass和基类Model的编写会比较复杂,但ORM的使用者用起来却异常简单。

现在,我们就按上面的接口来实现该ORM。

首先搞清楚我们应该做的工作:

  • 1. 既然要使得user能够通过继承Model,实现动态的创建类,那么必然要在ModelMetaclass中实现动态类的创建(比如能实现类Cutomer中的定义的attr属性们)
  • 2. 在基类Model中实现save等方法

首先来定义Field类,它负责保存数据库表的字段名和字段类型:

class Field(object):

    def __init__(self, name, column_type):
        self.name = name
        self.column_type = column_type

    def __str__(self):
        return "<%s:%s>" % (self.__class__.__name__, self.name)

Field的基础上,进一步定义各种类型的Field,比如StringFieldIntegerField等等:

class StringField(Field):

    def __init__(self, name):
        super(StringField, self).__init__(name, "varchar(100)")

class IntegerField(Field):

    def __init__(self, name):
        super(IntegerField, self).__init__(name, "bigint")

下一步,就是编写最复杂的ModelMetaclass了:

class ModelMetaclass(type):

    def __new__(cls, name, bases, attrs):
        if name=="Model":
            return type.__new__(cls, name, bases, attrs)
        print("Found model: %s" % name)
        mappings = dict()
        for k, v in attrs.items():
            if isinstance(v, Field):
                print("Found mapping: %s ==> %s" % (k, v))
                mappings[k] = v
        for k in mappings.keys():
            attrs.pop(k)
        attrs["__mappings__"] = mappings # 保存属性和列的映射关系
        attrs["__table__"] = name # 假设表名和类名一致
        return type.__new__(cls, name, bases, attrs)

当用户定义一个class Customer(Model)时,Python解释器首先在当前类Customer的定义中查找metaclass,如果没有找到,就继续在父类Model中查找metaclass,找到了,就使用Model中定义的metaclassModelMetaclass来创建Customer类,也就是说,metaclass可以隐式地继承到子类,但子类自己却感觉不到。

ModelMetaclass中,一共做了几件事情:

  1. 排除掉对Model类的修改(肯定要避免user通过调用ModelMetaclass来修改我们定义好的基类Model);

  2. 在当前类(比如Customer)中查找user定义的类(也就是例子中的Customer)的所有属性,如果找到一个Field属性,就把它保存到一个__mappings__的dict中,同时从类(Customer)的属性中删除该Field属性,否则,容易造成运行时错误(实例的属性会遮盖类的同名属性,也就是说保证Customer这个类里没有id,name等属性,从而不会影响Customer的实例的值; 这里之所以为了避免出现实例与类的同名属性,既避免运行时错误,是因为当出现类和实例的同名属性时,python会优先调用实例属性,如果没有的话则会调用类的属性,但是有时当实例属性不存在时,我们并不希望用户能调用到这个同名的类属性,比如当实例属性被删除时,再用相同的名字,就访问到了类属性了。因此才这么做);

  3. 把表名保存到__table__中,这里简化为表名默认为类名。

以及基类Model

class Model(dict, metaclass=ModelMetaclass):

    def __init__(self, **kw):
        super(Model, self).__init__(**kw)

    def __getattr__(self, key):
        try:
            return self[key]
        except KeyError:
            raise AttributeError(r""Model" object has no attribute "%s"" % key)

    def __setattr__(self, key, value):
        self[key] = value

    def save(self):
        fields = []
        params = []
        args = []
        for k, v in self.__mappings__.items():
            fields.append(v.name)
            params.append("?")
            args.append(getattr(self, k, None))
        sql = "insert into %s (%s) values (%s)" % (self.__table__, ",".join(fields), ",".join(params))
        print("SQL: %s" % sql)
        print("ARGS: %s" % str(args))

Model类中,就可以定义各种操作数据库的方法,比如save()delete()find()update等等。

我们实现了save()方法,把一个实例保存到数据库中。因为有表名,属性到字段的映射和属性值的集合,就可以构造出INSERT语句。

编写代码试试:

u = User(id=12345, name="Michael", email="test@orm.org", password="my-pwd")
u.save()

输出如下:

Found model: User
Found mapping: email ==> <StringField:email>
Found mapping: password ==> <StringField:password>
Found mapping: id ==> <IntegerField:uid>
Found mapping: name ==> <StringField:username>
SQL: insert into User (password,email,username,id) values (?,?,?,?)
ARGS: ["my-pwd", "test@orm.org", "Michael", 12345]

可以看到,save()方法已经打印出了可执行的SQL语句,以及参数列表,只需要真正连接到数据库,执行该SQL语句,就可以完成真正的功能。

 

最终,我们实现了关系型数据库的ORM框架,user可以通过继承Model类,把各种各样的表写成对象,进行操作。

总结一下,通过ORM的这个例子,我们可以看出Metaclass的作用总结起来就三点:

1.   拦截类的创建

2.   修改类

3.   返回修改之后的类

 

参考链接:

深刻理解Python中的元类(metaclass)

廖雪峰 Python 教程 使用元类

当前文章:http://hnhdqp.com/content/10-11/33466/content_8237038148.html

发布时间:2019-03-22 00:46:21

坐禅与静坐的区别何在 朱丹,爱自己的你好美 如何让“心理成长”帮助孩子更优秀 跑步=生小孩,是男人就该为女人跑一次步! 情话和性话,男女怎么说才受用? 出卖女人的是它们 家长“说教”是怎样毁人不倦的? 罗李华:天蝎座2016年运势

怎能没有光鲜好时候 一个好的人,不如一个对的人 五只猴子的故事 你的陨落,让世界失去了一位传奇---克里斯托弗·李 新西兰中小学教育有多好?多文化中文热 未来婆婆会不会成为扼杀感情的主要原因? 未婚夫要和别人结婚了 《鬼谷子七十二术》全解 请尊重一个姑娘的努力 长期坚持早起是什么感觉? 生命与爱 你敢不敢把手机借给陌生人? 当古体诗词遇见汉服 - 诗装 怎么确定家中的风水财位? 来得及 五只猴子的故事 错过天籁。。。 暖男是得不到性的 小升初:不学奥数,到底能不能上市重点?(下)

编辑:龙杜丁通

相关新闻

神东煤炭集团补连塔煤矿精检修抓安全保生产

2019-03-22 00:39:25

赣州承窘宗有限公司

榆林电网用电负荷、日用电量双创历史新高

2019-03-22 00:19:57

赣州沮弊副工程有限公司

一个陕北汉子的电影梦

2019-03-22 00:38:45

温州钥扑集团公司

《我是演说家》半决赛将播 乐嘉失控痛哭

2019-03-22 00:56:15

锦州用伎广告传媒有限公司

热门推荐

  • 腾讯:已完成发行50亿美元票据计划
  • Python socket
  • Nginx-动态路由升级版
  • 神秘大礼送不停 《创世神话》新服今日震撼开启
  • 《全域旅游广东造,智慧出行旅游+》系列电视节目摄影组走进揭西
  • 4月15日【周六】我们在这里挑人才,好工作c好人才最终会花落谁家呢?
  • 特朗普冒失喊话引担忧 中美被指或有开战风险
  • 四川茂县泥石流现场发现第15具罹难者遗体
  • 孙尧任教育部副部长 原任黑龙江省委常委(图)
  • 习总书记考察山西漫评:以革新成效迎接十九大
  • 河北新闻网版权所有 本站点信息未经允许不得复制或镜像 法律顾问:石油美元是如何诞生的? 你是那个内心强大的人吗?
  • 父母的压力会“传染”给孩子 copyright ? 2000 - 2016
  • 新闻热线:0311-67563366 广告热线:0311-67562966 新闻投诉:0311-67562994
  • 冀ICP备 09047539号-1 | 互联网新闻信息服务许可证编号:1312006002
  • 广播电视节目制作经营许可证(冀)字第101号|信息网络传播视听节目许可证0311618号
  • 名字,孩子成长中不能忽视 女生最爱问的七个问题的标准答案 2011成长营:参加国际大赛的经历 家居风水好,财运滚滚来(家居风水简单学)