Python进阶开发之元类编程

Posted Python编程时光

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Python进阶开发之元类编程相关的知识,希望对你有一定的参考价值。

系列文章

第一章 元类编程,已完成 ;

本文目录

类是如何产生的如何使用type创建类理解什么是元类使用元类的意义元类实战:ORM

. 类是如何产生的

类是如何产生?这个问题肯定很傻。实则不然,很多人只知道使用继承的表面形式来创建一个类,却不知道其内部真正的创建是由type来创建的。

type?这不是判断对象类型的函数吗?

是的,type通常用法就是用来判断对象的类型。但除此之外,他最大的用途是用来动态创建类。当Python扫描到class的语法的时候,就会调用type函数进行类的创建。

. 如何使用type创建类

首先,type()需要接收三个参数

11. 类的名称,若不指定,也要传入空字符串:""
22. 父类,注意以tuple的形式传入,若没有父类也要传入空tuple:(),默认继承object
33. 绑定的方法或属性,注意以dict的形式传入

来看个例子

 1# 准备一个基类(父类)
2class BaseClass:
3 def talk(self):
4 print("i am people")
5
6# 准备一个方法
7def say(self):
8 print("hello")
9
10# 使用type来创建User类
11User = type("User", (BaseClass, ), {"name":"user", "say":say})

. 理解什么是元类

什么是类?可能谁都知道,类就是用来创建对象的「模板」。

那什么是元类呢?一句话通俗来说,元类就是创建类的「模板」。

为什么type能用来创建类?因为它本身是一个元类。使用元类创建类,那就合理了。

type是Python在背后用来创建所有类的元类,我们熟知的类的始祖 object 也是由type创建的。更有甚者,连type自己也是由type自己创建的,这就过份了。

1>>> type(type)
2<class \'type\'>
3>>> type(object
4<class \'type\'>
5>>> type(int)
6<class \'type\'>
7>>> type(str)
8<class \'type\'>

如果要形象的来理解的话,就看下面这三行话。

1str:用来创建字符串对象的类。
2int:是用来创建整数对象的类。
3type:是用来创建类对象的类。

反过来看

1一个实例的类型,是类
2一个类的类型,是元类
3一个元类的类型,是type

来看下实例

 1# Python3.7
2>>> class MetaPerson(type):
3... pass
4...
5>>> class Person(metaclass=MetaPerson):
6... pass
7...
8>>> Tom = Person()
9>>> print(type(Tom))
10<class \'__main__.Person\'>
11>>> print(type(Tom.__class__))
12<class \'__main__.MetaPerson\'>
13>>> print(type(Tom.__class__.__class__))
14<class \'type\'>

上面是一个简单的示例。

下面看一个稍微完整的

 1# 注意要从type继承
2class BaseClass(type):
3 def __new__(cls, *args, **kwargs):
4 print("in BaseClass")
5 return super().__new__(cls, *args, **kwargs)
6
7class User(metaclass=BaseClass):
8 def __init__(self, name):
9 self.name = name
10
11user = User("wangbm")

. 使用元类的意义

正常情况下,我们都不会使用到元类。但是这并不意味着,它不重要。假如某一天,我们需要写一个框架,很有可能就需要用到元类。

但是,为什么要用它呢?不要它会怎样?

经过我的总结,元类的作用过程如下

  1. 拦截类的创建
  2. 拦截下后,进行修改
  3. 修改完后,返回修改后的类

很明显,使用元类,是要对类进行定制修改。使用元类来动态生成元类的实例,而99%的开发人员是不需要动态修改类的,因为这应该是框架才需要考虑的事。

但是,这样说,你一定不会服气,到底元类用来干什么?其实元类的作用就是创建API,一个最典型的应用是 Django ORM。

. 元类实战:ORM

使用过Django ORM的人都知道,有了ORM,使得我们操作数据库,变得异常简单。

ORM的一个类(User),就对应数据库中的一张表。id,name,email,password 就是字段。

1class User(BaseModel):
2 id = IntField(\'id\')
3 name = StrField(\'username\')
4 email = StrField(\'email\')
5 password = StrField(\'password\')
6
7 class Meta:
8 db_table = "user"

如果我们要插入一条数据,我们只需这样做

1# 实例化成一条记录
2u = User(id=20180424, name="xiaoming",
3 email="xiaoming@163.com", password="abc123")
4
5# 保存这条记录
6u.save()

通常用户层面,只需要懂应用,就像上面这样操作就可以了。

但是今天我并不是来教大家如何使用ORM,我们是用来探究ORM内部究竟是如何实现的。我们也可以自己写一个简易的ORM。

从上面的User类中,我们看到StrFieldIntField,从字段意思上看,我们很容易看出这代表两个字段类型。字段名分别是id,username,email,password

StrFieldIntField在这里的用法,叫做属性描述符,如果对这个不了解的可以查看文章底部的参考文章,也是我写的。
简单来说呢,属性描述符可以实现对属性值的类型,范围等一切做约束,意思就是说变量id只能是int类型,变量name只能是str类型,否则将会抛出异常。

那如何实现这两个属性描述符呢?请看代码。

 1import numbers
2
3class Field:
4 pass
5
6class IntField(Field):
7 def __init__(self, name):
8 self.name = name
9 self._value = None
10
11 def __get__(self, instance, owner):
12 return self._value
13
Python之元类详解

Python之元类ORM

Python之元类

Python之元类详细解析

Python之Metaclass详解,Python之元类

面向对象之元类