这是一个在看开源代码时配到了单例元类写法后的一个原理贴,可以帮你彻底弄清元类、以及
__new__
和__call__
,相信我,看完你绝对会有收获。首先,关于元类的结论说在前头,先有个印象——小总结:
- 现在我们能知道为什么元类必须继承type了:因为我们实例化对象
Foo(xxx)
时调用了type.__call__
,而type.__call__
又会调用type.__new__
因此如果type子类重写实现了__new__
(返回的类实例对象的类型作控制)、__call__
(对实例化的流程做控制),则可以对类对象的类型和类属性起到自定义的功能,而重写就必须继承type=>需要元类必须继承type- 所以按照上述的逻辑,如果定义了一个元类让自定义类用的话
class Foo(metaclass=MyMetaClass)
,在其实例化过程中Foo()
会直接调用重写后的MyMetaClass.__call__
,而只要记住在MyMetaClass.__call__
中使用到return super(Singleton, cls).__call__(*args, **kwargs)
就可以把type.__call__
生成的实例返回啦。所以这也是为什么编写元类,一般都是继承了type,然后根据想控制实例化流程就重写__call__
方法,想添加属性就重写__new__
方法就行了。- ★元类产生影响的时间点是在实例化的时候
看开源代码时,看到了下面一段代码,于是对withMetaclass产生了好奇,经过了解发现其作用是six对python2和python3使用元类兼容的写法。
1 | # Python2和3兼容使用元类写法 |
因此,上述代码在Python3中相当于
1 | # Python3元类使用写法 |
那么,问题来了,withMetaclass到底是怎么实现兼容的呢?下面是其实现代码
1 | def withMetaclass(meta, *bases): |
可以看到其中出现了不少我们很少看到的使用方法。接下来我们就仔细的学习上述写法为什么可以成功。
元类使用可以参考:Python3 元类(metaclass)
预置知识:type和object
object 和 type的关系很像鸡和蛋的关系,先有object还是先有type没法说,obejct和type是共生的关系,必须同时出现的。
记住一点:在Python里面,所有的东西都是对象的概念,即包括类(类是type的实例对象)
最重要的两点
- object类是所有类的超类(也是type类的父类)
- type是所有类的类(类型,所有类都是type的实例对象,object类型也是type的实例对象;type 创建的对象拥有创建对象的能力(也就是类))–>是所有类的元类
此外:
- type是所有元类的父亲。我们可以通过继承type来创建元类(通过重写
type.__new__
和type.__call__
来拦截自定义类的创建过程)。 - object是所有类的父亲。
- 实例是对象关系链的末端,不能再被子类化和实例化。
了解到这些关键的点后,我们继续看代码中出现的一些内容:
__new__
__new__()
是一种负责创建类实例的静态方法,它无需使用 staticmethod 装饰器修饰,且该方法会优先__init__()
初始化方法被调用。
__new__()
通常会返回该类的一个实例,但有时也可能会返回其他类的实例,其super().__new__(cls)
中会调用object.__init__
来Create and return a new object.
,因此我们可以通过改写子类的__new__
可以添加一些逻辑来控制实例的产生,然后再通过super().__new__(cls)
来生成一个instance并返回。
1 | class demoClass: |
Q:什么情况下重写类的__new__()
呢?答案很简单,在__init__()
不够用的时候。
__new__()
通常会返回该类的一个实例,但有时也可能会返回其他类的实例,如果发生了这种情况,则会跳过对 __init__()
方法的调用。而在某些情况下(比如需要修改不可变类实例(Python 的某些内置类型)的创建行为),利用这一点会事半功倍。比如:http://c.biancheng.net/view/5484.html,对 Python 不可变的内置类型(如 int、str、float 等)进行了子类化,这是因为一旦创建了这样不可变的对象实例,就无法在__init__()
方法中对其进行修改。
注:由于 __new__()
不限于返回同一个类的实例,所以很容易被滥用,不负责任地使用这种方法可能会对代码有害,所以要谨慎使用。
MetaClass元类
承接上文
__new__
,Python中大量使用__new__()
方法且合理的地方,就是 MetaClass 元类。MetaClass元类,并不是某一个类的名字,它是一个概念,是一种Python的思想。当然其本质也是一个类,但和普通类的用法不同,它可以对类内部的定义(包括类属性和类方法)进行动态的修改。可以这么说,使用元类的主要目的就是为了实现在创建类时,能够动态地改变类中定义的属性或者方法。其可以将创建对象的过程拦截下来,从而对这个对象进行自定义(这个需要类继承type,与前文继承object的做区别)。
明确一点:元类可以理解成是自定义类继承的父类(从兼容写法中也能看出),但元类的特点是不会出现在自定义类的继承关系(
__mro__
)之中
举个例子,根据实际场景的需要,我们要为多个类添加一个 name 属性和一个 say() 方法。显然有多种方法可以实现,但其中一种方法就是使用 MetaClass 元类。
1 | # 定义一个元类,继承type。因为只有继承type才能通过重写__new__来拦截创建过程 |
可以看到,在创建类时,通过在标注父类的同时指定元类(格式为metaclass=元类名
),则当 Python 解释器在创建该类实例时,FirstMetaClass(type)
元类中的__new__
方法就会被调用,其中bases和attrs能拿到自定义类的参数,从而实现动态修改类属性或者类方法的目的。
元类和父类的区别:
在定义子类的时候,我们有两个选择:①是传需要继承的父类;②自定义的元类。
- 父类是子类的模板,子类的功能是跟父类紧耦合的,子类和父类一般是一一对应的
- 元类是子类的修饰器,可以为该子类和其他子类都添加自定义功能,并且不在继承关系中(
Class.__mro__
查看),子类和元类是一对多的关系。元类并不是特地为某个子类服务的
1 | class TestMeta3(type): |
在定义的时候,发现竟然有输出。因为定义的时候,python解释器会在当前类中查找metaclass[3],如果找到了,就使用该metaclass创建Eg3类。所以打印出来的name、bases、attrs都和Eg3有关。
with_metaclass
由于python2和python3中元类使用方法的不同,我们需要使用一种兼容的方式[1],如下所示:
1 | def withMetaclass(meta, *bases): |
with_metaclass
返回的临时类中,本身无任何属性,但包含了元类和基类的所有信息,并在下一步定义类时将所有信息解包出来[1]。
type
动态创建类
- type() 函数属于 Python 内置函数,通常用来查看某个变量的具体类型。
type(obj)
- 其实,type() 函数还有一个更高级的用法,即创建一个自定义类型(也就是创建一个类)。
type(name, bases, dict)
:其中 name 表示类的名称;bases 表示一个元组,其中存储的是该类的父类;dict 表示一个字典,用于表示类内定义的属性或者方法。
实际上type(name, bases, dict)
是调用了type类的type.__init__(cls, what, bases=None, dict=None)
方法,创建了一个type的实例(类类型就是一个type实例),类型是<class 'type'>
<class ‘type’>是所有类型的类型。<class ‘object’>也是所有对象的超类(除了它自己,包括type)
▲. 此外type还有type.__new__(*args, **kwargs)
,其作用是Create and return a new object.
,可以写成type.__new__(ClassTpye, name, base, dicts)
,但ClassType必须是type的子类。会返回一个跟ClassType有关系的新类型
通过元类创建单例类
现在让我们正式看,我在开源代码里看到的内容:
1 | # 注意这边继承了type, 所以下面的__call__是重写type的__call__,即创建实例的方法 |
注:类也是对象,是元类的对象,即我们实例化一个类时,调用其元类的__call__(cls, *args, **kwargs)
方法进行创建对象。
__call__
一个非常特殊的实例方法,即
__call__()
。该方法的功能是在类中重载了对象的 () 运算符,使得类实例对象可以像调用普通函数那样,以“对象名()”的形式使用。
实际上,如果不重写__call__
的话,Class.__call__(*args, **kwargs)
还承担着产生类实例的功能(会调用父类(可以通过Class.__class__
来查看父类)的type.__call__
其会返回一个实例)
案例一:
1 | # 默认继承的是object, 而不是type |
Q:我们在实例化一个对象的时候f = Foo(1, y=2)
,可以发现在__init__()
中并没有返回实例,但调用Foo(1, y=2)
确实返回了一个对象,而且,__init__
预期一个self
参数,但是当我们调用Foo(1, y=2)
时这里并没有这个参数。那么类实例化的过程到底是怎么样的呢?
A:构造顺序——理解python的类实例化
首先明确一点,Python中的类也是对象!类、函数、方法以及实例都是对象——类类型是type的对象,并且无论何时你将一对括号放在它们的名字后面时,就会调用type.__call__()
方法。为什么呢?因为type是类型的父类
1 | Foo.__class__ |
所以Foo
是类型type
的一个对象,并且调用type类的__call__(self, *args, **kwargs)
返回一个Foo
类的对象。让我们看下type
中的__call__
方法是什么样的。这个方法相当的复杂,但是我们将其C代码转成Python代码,并尝试尽量简化它,结果如下。
1 | class type(object): |
可见__new__
方法为对象分配了内存空间,构建它为一个“空"对象然后__init__
方法被调用来初始化它。
那我们定义了一个具体类来讲解这个过程。首先明确一点:Foo相对于产生了一个type实例化对象
1 | class Foo(object): |
获得实例化对象**Foo(*args, **kwargs)
也可以看作是type对象()
即调用了type中()运算符的触发的函数type.__call__
从而创建一个Foo的实例**
- 至于
type.__call__
发生了什么就是上面抽象代码中介绍的那般,调用type.__new__(Foo, *args, **kwargs)
然后返回一个对象实例obj。 obj
随后通过调用obj.__init__(*args, **kwargs)
被初始化。obj
被type.__call__
中返回。
▲注意:Foo.__call__
重载的是foo对象
的()运算符,而Foo()
实例化foo对象,则执行的是type对象
的()运算符。
小总结:
- 现在我们能知道为什么元类必须继承type了:因为我们实例化对象
Foo(xxx)
时调用了type.__call__
,而type.__call__
又会调用type.__new__
因此如果type子类重写实现了__new__
(返回的类实例对象的类型作控制)、__call__
(对实例化的流程做控制),则可以对类对象的类型和类属性起到自定义的功能,而重写就必须继承type=>需要元类必须继承type - 所以按照上述的逻辑,如果定义了一个元类让自定义类用的话
class Foo(metaclass=MyMetaClass)
,在其实例化过程中Foo()
会直接调用重写后的MyMetaClass.__call__
,而只要记住在MyMetaClass.__call__
中使用到return super(Singleton, cls).__call__(*args, **kwargs)
就可以把type.__call__
生成的实例返回啦。所以这也是为什么编写元类,一般都是继承了type,然后根据想控制实例化流程就重写__call__
方法,想添加属性就重写__new__
方法就行了。 - ★元类产生影响的时间点是在实例化的时候
注意点:元类继承了type,所以实例化元类是在产生一个类类型,就要以type创建类类型的参数去产生。而元类的使用一般都是自定义类class MyClass(metaclass=元类)
,然后实例化自定义类MyClass(xxx)
总结:看完上述知识点后,我们能知道为什么withclass能起到metaclass的作用(类的__mro__
中不出现指定的元类)了:
- 首先分析流程:
return type.__new__(Metaclass)
返回了一个类型供自定义类继承,由于MetaClass继承的是真正的元类(元类都继承type),所以在自定义类实例化的时候会被Metaclass的__new__
方法拦截,在MetaClass.__new__
里return了一个自定义实例,并把对象加入到了Singleton字典中了。 - 其次讲解为什么MetaClass中没有MetaClass:因为根据
__new__
知识点中讲到的,__new__
控制了实例产生,return type.__new__(Metaclass)
中创建了Metaclass
,但其在__new__
中返回的并不是MetaClass,因此__mro__
中不会出现Metaclass
- 最后还要讲讲Singleton中的执行逻辑:
1 | class Singleton(type): |
Author: Mrli
Link: https://nymrli.top/2022/04/05/理了一天彻底弄懂元类——分享给你一起弄懂/
Copyright: All articles in this blog are licensed under CC BY-NC-SA 3.0 unless stating additionally.