我的编程空间,编程开发者的网络收藏夹
学习永远不晚

由ORM谈Python元类

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

由ORM谈Python元类

本文主要介绍了什么是类,如何动态的创建类,什么是元类,类的_metaclass_属性以及如何自己实现ORM。

在谈Python元类之前,我们先来看看ORM是什么:

ORM 为Object Relational Mapping 的简称,中文意思为对象关系映射,ORM的出现,是为了解决面向对象和关系数据库之间不相匹配而出现的技术,目的是使用面向对象的思想来实现对数据库中表的操作,从而屏蔽掉底层的SQL操作。

简单了解了ORM的概念,许多同学可能会想了解ORM是怎么实现的,以下代码摘自django项目ORM的部分实现代码:

subclass_exception'__module__': module}
   if attached_to is not None:
           __reduce__return __setstate__'__reduce__'] = __reduce__         class_dict['__setstate__'] = return type>>> class MyClass():
...     pass
...
>>> type(MyClass) <type 'classobj'
>>>> type(MyClass()) <type 'instance'>

同时,它也具有对象的全部特征,如你可以对他进行赋值,拷贝,增加属性等的操作,如下示例:

>>> class MyClass():
...     pass ...
>>> print MyClass __main__.MyClass
>>> a = MyClass
>>> print a __main__.MyClass
>>> MyClass.__dict__
{'__module__': '__main__', '__doc__': None}
>>> MyClass.a = 100
>>> MyClass.__dict__
{'a': 100, '__module__': '__main__', '__doc__': None}
>>> >>> MyClass.a = 200
>>> MyClass.a200
>>>
如何动态的创建类?

最简单的方法,就是采用如下方法:

def create_class(class_name):     if class_name == "MyClass":
       class MyClass():             pass         return MyClass
   else:
       class OtherClasss():             pass         return OtherClasss

但是此类方法,不能称之为真正的动态,因为只能支持有限的类的创建,那有什么方法,能真正实现动态创建一些类呢? 所以,这里就不得不提到关键词type了,首先,我们大多数人谈起type,首先想到的是,type不就是用来测试变量类型的嘛,如下:

>>> type1)<type 'int'
>>>> print type"1")<type 'str'
>>>> class AtypetypeMyClass = MetaClass()
MyObject = MyClass()

而type,就是元类,于是创建一个类,就可以使用如下这种方式:

MyClass = type("MyClass", (), {})
MyObject = MyClass()
类的_metaclass_属性

当我们创建一个类的时候为其添加__metaclass__属性:

class Foo(object):     __metaclass__ = something… […]

那么,Python就会采用你指定的方法,来创建这个类,否则,会使用type来创建这个类,下面演示如何自定义的指定metaclass属性:

lower_attrfor name, val in class_attr.items():
       if name.startswith('__'):            lower_attr[name] = val
       else:            lower_attr[name.lower()] = return typeMyClss'A'     b = 'b'

my_class = MyClss()
hasattr'A') # False print hasattr'b') # True print hasattr'a') # True
如何自己实现ORM?

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

class Field(object):     def __init__(self, name, column_type):         self.name = name         self.column_type = column_type

在Field的基础上,进一步定义各种具体类型的Field,比如StringField,IntegerField等等:

class StringField(Field):     def __init__(self, name, max_length):         super(StringField, self).__init__(name, 'varchar(%d)' % max_length)  class IntegerField(Field):     def __init__(self, name):         super(IntegerField, self).__init__(name, 'int')

下一步,就是编写最复杂的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):                 mappings[k] = v         for k in mappings.keys():             attrs.pop(k)         attrs['__mappings__'] = mappings         attrs['__table__'] = name         return type.__new__(cls, name, bases, attrs)

以及基类Model:

class Model(dict, metaclass=ModelMetaclass):

   __init__super__init____getattr__try:
           return self[key]         except KeyError:             raise AttributeError"'Model' object has no attribute '%s'" % key)      def __setattr__savefor 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))

创建User类,并调用save方法,保存:

class User(Model):     id = IntegerField('id')     name = StringField('username', max_length=48)     email = StringField('email', max_length=128)     password = StringField('password', max_length=24)  # 创建一个实例: u = User(id=12345, name='xiaoming', email='xiaoming@xiaomi.com', password='test') # 保存到数据库: u.save()

上面,我们就简单实现了一个ORM。

参考链接

stackoverflow:

https://stackoverflow.com/questions/100003/what-are-metaclasses-in-python 

廖雪峰的官方网站:

https://www.liaoxuefeng.com/wiki/001374738125095c955c1e6d8bb493182103fac9270762a000/001386820064557c69858840b4c48d2b8411bc2ea9099ba000

由ORM谈Python元类

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

由ORM谈Python元类

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

由ORM谈Python元类

本文主要介绍了什么是类,如何动态的创建类,什么是元类,类的_metaclass_属性以及如何自己实现ORM。在谈Python元类之前,我们先来看看ORM是什么:ORM 为Object Relational Mapping 的简称,中文意思为
2023-06-02

Python之元类ORM

什么是元类在Python中一切皆对象,类也是一个对象,实例对象由类创建出来的,类是由元类创建出来的。简而言之,用来创建类的类就叫元类(metaclass)。 函数type其实就是一个元类,type就是Python在背后用来创建所有类的元类。
2023-01-31

用 Python 元类的特性实现 ORM 框架

目录ORM是什么实现ORM中的insert功能完善对数据类型的检测抽取到基类中添加数据库驱动执行sql语句添加数据库驱动执行sql语句测试功能准备数据库创建模型类测试源代码ORM是什么O是 object,也就 类对象 的意思,R是 rela
2022-06-02

Python元类编程实现一个简单的ORM

本文主要介绍了Python元类编程实现一个简单的ORM,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-03-06

如何使用Python元类特性实现ORM框架

这篇文章主要介绍了如何使用Python元类特性实现ORM框架,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。ORM是什么O是 object,也就 类对象 的意思,R是 rela
2023-06-15

怎么使用Python元类编程实现一个简单的ORM

这篇文章主要讲解了“怎么使用Python元类编程实现一个简单的ORM”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么使用Python元类编程实现一个简单的ORM”吧!概述什么是ORM?  
2023-07-05

简单谈谈Python中的元祖(Tuple)和字典(Dict)

前言 本文记录了对于Python的数据类型中元祖(Tuple)和字典(Dict)的一些认识,以及部分内置方法的介绍。下面话不多说,来看看详细的介绍吧。 元祖 Tuple 特点:元祖内的数据不可变 一个元素的定义:T = (1,)>>> T=
2022-06-04

浅谈Python的文件类型

Python的文件类型主要分为3种:源代码(source file)、字节码(byte-code file)、优化的字节码(optimized file)。这些代码都可以直接运行,不需要编译或者连接。这正是Python语言的特性,Pytho
2022-06-04

python中的元类

目录 元类 什么是元类 自定义元类 其他 元类什么是元类我们创建一个类目的是为了创建该类的实例对象,而元类就是用来创建类的
2023-01-30

浅谈Python响应式类库RxPy

目录一、基本概念1.1、Observable和Observer(可观察对象和观察者)1.2、Operator(操作符)1.3、Single(单例)1.4、Subject(主体)1.5、Scheduler(调度器)1.6、Observer和O
2022-06-02

Python黑魔法:元类

Python黑魔法:元类术语“元编程”指的是程序具有编写或操纵其自身作为它们资料的潜力。Python支持称为元类的类的元编程。元类是一个深奥的面向对象编程(OOP)概念,隐藏在几乎所有的Python代码之后。无论你是否意识到它的存在,你都一
2023-01-31

Python 5.5 使用元类

使用元类type()动态语言和静态语言最大的不同,就是函数和类的定义,不是编译时创建的,而是运行时动态创建的。比方说我们定义一个Hello的class,就写一个hello.py模块:class Hello(object):    def h
2023-01-31

浅谈Python由__dict__和dir()引发的一些思考

关于__dict__和dir()的区别和作用请参考这篇文章: 基于Python __dict__与dir()的区别详解 说下我当时遇到的问题:class Demo:def __init__(self, name, age):self.nam
2022-06-04

简单谈谈Python中的几种常见的数据类型

计算机顾名思义就是可以做数学计算的机器,因此,计算机程序理所当然地可以处理各种数值。但是,计算机能处理的远不止数值,还可以处理文本、图形、音频、视频、网页等各种各样的数据,不同的数据,需要定义不同的数据类型。在Python中,能够直接处理的
2022-06-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录