python中元类的__call__方法的作用

本文探讨了Python中元类的概念,强调元类是类的类,类似于类是实例的类。元类的`__new__`方法用于创建类,而`__call__`方法则会在实例化时被调用,可以拦截类的构造过程。通过示例代码展示了`__call__`如何影响类实例的创建,并指出如果`__call__`不返回对象,将导致实例化失败。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

元类是类的类,元类之于类就相当于类之于实例。
元类的new方法会创建一个类并返回,就像类的new方法会创建一个实例并返回一样。
元类中其他方法的定义类似于类中方法的定义,例如:
class Meta(type): 
    def __new__(cls, name, bases, dct):  # cls为元类Meta
        return type.__new__(cls, name, bases, dct)

    def foo(cls, *args, **kwargs):  # cls为元类创建的类
        pass

    def __call__(cls, *args, **kwargs):  # cls为元类创建的类
        pass

元类中有一个特殊的方法__call__,这个方法会阶段类的__new____init__

__call__ 应该返回实例,和类的__new__方法返回的一样。

下面看几个例子:

class Meta(type):
    def __new__(cls, name, bases, dct):
        print("calling Meta's __new__", cls)
        return type.__new__(cls, name, bases, dct)

    def __call__(cls, *args, **kwargs):
        print("calling Meta's __call__", cls)
        i = cls.__new__(cls)
        i.__init__(*args, **kwargs)
        return i

class A(object):
    __metaclass__ = Meta

    def __new__(cls, *args, **kwargs):
        print("calling A's __new__")
        return object.__new__(cls)

    def __init__(self, *args, **kwargs):
        print("calling A's __init__")

a = A()
print("a is", a)
运行结果
calling Meta's __new__ <class '__main__.Meta'>
calling Meta's __call__ <class '__main__.A'>
calling A's __new__
calling A's __init__
a is <__main__.A object at 0x7faadaf02390>
此时,还不太能看出来Meta.call拦截了A.__new__A.__init__,只能看出,__call__会先于__new____init__调用。(其实可以看出,如果没有拦截发生,calling A's __new__calling A's __init__会输出两次)
为了更清楚的看出拦截行为,我们更改一下类的定义:
class Meta(type):
    def __new__(cls, name, bases, dct):
        print("calling Meta's __new__", cls)
        return type.__new__(cls, name, bases, dct)

    def __call__(cls, *args, **kwargs):
        print("calling Meta's __call__", cls)
        i = object.__new__(cls)
        i.__init__(*args, **kwargs)
        return i

class A(object):
    __metaclass__ = Meta

    def __new__(cls, *args, **kwargs):
        print("calling A's __new__")
        return object.__new__(cls)

    def __init__(self, *args, **kwargs):
        print("calling A's __init__")
输出结果:
calling Meta's __new__ <class '__main__.Meta'>
calling Meta's __call__ <class '__main__.A'>
calling A's __init__
a is <__main__.A object at 0x7fd6b122c3d0>
可以看出,A.__new__没有被调用,__call__返回的即为类的实例对象。
如果去掉__call__中的__init__调用,上面输出结果中就不会出现calling A's __init__,由此可以确定,__call__拦截了__new____init__

注意:

如果元类中定义了__call__,此方法必须返回一个对象,否则类的实例化就不会起作用。(实例化得到的结果为__call__的返回值)

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值