python 元类的call总结_python中元类的__call__方法的作用

元类是类的类,元类之于类就相当于类之于实例。

元类的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]联系我们处理,如需商业授权请联系原作者/原网站。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值