美文网首页Python之web开发Pythoner集中营程序员
python中元类的__call__方法的作用

python中元类的__call__方法的作用

作者: 不_一 | 来源:发表于2018-01-13 11:14 被阅读56次

元类是类的类,元类之于类就相当于类之于实例。
元类的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__
<__main__.Foo object at 0x10ae85860>

此时,还不太能看出来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__
<__main__.Foo object at 0x106004828>

可以看出,SingletonType.__new__没有被调用,__call__返回的即为类的实例对象。

如果注释掉掉__call__中的__init__调用,上面输出结果中就不会出现Foo __init__,由此可以确定,__call__拦截了__new____init__

强调

  1. 如果元类中定义了__call__,此方法必须返回一个对象,否则类的实例化就不会起作用。(实例化得到的结果为__call__的返回值)
  2. 如果元类的__call__中返回type.__call__(cls, *args, **kwargs),type创建的对象,里面会调用Foo__new__方法,和__init__方法
  3. 第2项一定要改一下上面的代码进行验证.谨记!!!谨记!!!谨记!!!!!!!

相关文章

  • python中元类的__call__方法的作用

    元类是类的类,元类之于类就相当于类之于实例。元类的new方法会创建一个类并返回,就像类的new方法会创建一个实例并...

  • Python 单例模式

    元类方法__call__ 可以这样使用: 如下: 调用__call__实现python 单例模式 元类继承: 测试:

  • 名为__call__的特殊方法

    通过名为__call__的特殊方法,可以使类的实例能够像普通的Python函数那样得到调用。 current = ...

  • 28.Python之内置的魔术方法

    Python之内置的魔术方法 __call__方法在调用对象时(实例化的对象加括号),会自动触发类中的__call...

  • Python装饰器(5)类装饰器

    内容纯属个人理解,不对之处,欢迎指正。 __call__ 在Python中,具有__call__方法的对象才可以被...

  • python中__call__方法解析

    __call__方法是python的魔术方法,就是增加一些特殊功能,我们来看看__call__代表了什么意思?py...

  • python中的functools模块

    functools模块可以作用于所有的可以被调用的对象,包括函数 定义了__call__方法的类等 1 funct...

  • py_27 类装饰器

    类装饰器 实例() 其实是调用__call__方法 因此可以 重写__call__方法,从而给函数添加装饰器 1....

  • Python 的 __call__ 方法

    示例 通过上面的小例子可以看到,__call__ 方法的作用就是让类的实例可以像函数一样被调用(called)。可...

  • Collections工具类的使用

    作用:操作Collection和Map的工具类 常用方法:reverse(List):反转 List 中元素的顺序...

网友评论

    本文标题:python中元类的__call__方法的作用

    本文链接:https://www.haomeiwen.com/subject/buccoxtx.html