元类是类的类,元类之于类就相当于类之于实例。
元类的new方法会创建一个类并返回,就像类的new方法会创建一个实例并返回一样。
元类中其他方法的定义类似于类中方法的定义,例如:
class Meta(type):
def __new__(cls, name, bases, dct): # cls为元类Meta
return type.__new__(cls, name, bases, dct)
def __init__(cls, *args, **kwargs): # cls为元类创建的类
pass
def __call__(cls, *args, **kwargs): # cls为元类创建的类
pass
元类中有一个特殊的方法 __call__
,这个方法会截断类的 __new__
和 __init__
方法,阻止其执行
__call__
应该返回实例,和类的 __new__
方法返回的一样。
下面看几个例子:
class SingletonType(type):
def __init__(cls, *args, **kwargs):
print('元类__init__')
super(SingletonType, cls).__init__(*args, **kwargs)
def __call__(cls, *args, **kwargs):
print('元类__call__')
obj = cls.__new__(cls, *args, **kwargs)
cls.__init__(obj, *args, **kwargs) # Foo.__init__(obj)
return obj
class Foo(metaclass=SingletonType):
def __init__(self, name):
print("Foo __init__")
self.name = name
def __new__(cls, *args, **kwargs):
print('Foo __new__')
return object.__new__(cls)
obj = Foo('name')
print(obj)
运行结果:
元类__init__
元类__call__
Foo __new__
Foo __init__
此时,还不太能看出来 SingletonType.__call__
拦截了 Foo.__new__
和Foo. init
,只能看出, __call__
会先于 __new__
和 __init__
调用。(其实可以看出,如果没有拦截发生, Foo __new__
, Foo __init__
会输出两次)
为了更清楚的看出拦截行为,我们更改一下类的定义:
class SingletonType(type):
def __init__(cls, *args, **kwargs):
print('元类__init__')
super(SingletonType, cls).__init__(*args, **kwargs)
def __call__(cls, *args, **kwargs):
print('元类__call__')
# obj = cls.__new__(cls, *args, **kwargs)
obj = object.__new__(cls)
cls.__init__(obj, *args, **kwargs) # Foo.__init__(obj)
return obj
class Foo(metaclass=SingletonType):
def __init__(self, name):
print("Foo __init__")
self.name = name
def __new__(cls, *args, **kwargs):
print('Foo __new__')
return object.__new__(cls)
obj = Foo('name')
print(obj)
结果:
元类__init__
元类__call__
Foo __init__
可以看出, SingletonType.__new__
没有被调用, __call__
返回的即为类的实例对象。
如果注释掉掉 __call__
中的 __init__
调用,上面输出结果中就不会出现 Foo __init__
,由此可以确定, __call__
拦截了 __new__
和 __init__
强调
如果元类中定义了 call
,此方法必须返回一个对象,否则类的实例化就不会起作用。(实例化得到的结果为 call
的返回值)
如果元类的 __call__
中返回 type.__call__(cls, *args, **kwargs)
,type创建的对象,里面会调用 Foo
的 __new__
方法,和 __init__
方法
第2项一定要改一下上面的代码进行验证.谨记!!!谨记!!!谨记!!!!!!!
注意:本文来自简书。本站无法对本文内容的真实性、完整性、及时性、原创性提供任何保证,请您自行验证核实并承担相关的风险与后果!
CoLaBug.com遵循[CC BY-SA 4.0]分享并保持客观立场,本站不承担此类作品侵权行为的直接责任及连带责任。您有版权、意见、投诉等问题,请通过[eMail]联系我们处理,如需商业授权请联系原作者/原网站。