python面试的100题(15)

41.super函数的具体用法和场景

为了调用父类(超类)的一个方法,可使用 super() 函数,好比:html

class A:
    def spam(self):
        print('A.spam')

class B(A):
    def spam(self):
        print('B.spam')
        super().spam()  # Call parent spam()

super() 函数的一个常见用法是在 __init__() 方法中确保父类被正确的初始化了:python

class A:
    def __init__(self):
        self.x = 0

class B(A):
    def __init__(self):
        super().__init__()
        self.y = 1

super() 的另一个常见用法出如今覆盖Python特殊方法的代码中,好比:算法

class Proxy:
    def __init__(self, obj):
        self._obj = obj

    # Delegate attribute lookup to internal obj
    def __getattr__(self, name):
        return getattr(self._obj, name)

    # Delegate attribute assignment
    def __setattr__(self, name, value):
        if name.startswith('_'):
            super().__setattr__(name, value) # Call original __setattr__
        else:
            setattr(self._obj, name, value)

在上面代码中,__setattr__() 的实现包含一个名字检查。 若是某个属性名如下划线(_)开头,就经过 super() 调用原始的 __setattr__() , 不然的话就委派给内部的代理对象 self._obj 去处理。 这看上去有点意思,由于就算没有显式的指明某个类的父类, super() 仍然能够有效的工做。ide

讨论

实际上,你们对于在Python中如何正确使用 super() 函数广泛知之甚少。 你有时候会看到像下面这样直接调用父类的一个方法:wordpress

class Base:
    def __init__(self):
        print('Base.__init__')

class A(Base):
    def __init__(self):
        Base.__init__(self)
        print('A.__init__')

尽管对于大部分代码而言这么作没什么问题,可是在更复杂的涉及到多继承的代码中就有可能致使很奇怪的问题发生。 好比,考虑以下的状况:函数

class Base:
    def __init__(self):
        print('Base.__init__')

class A(Base):
    def __init__(self):
        Base.__init__(self)
        print('A.__init__')

class B(Base):
    def __init__(self):
        Base.__init__(self)
        print('B.__init__')

class C(A,B):
    def __init__(self):
        A.__init__(self)
        B.__init__(self)
        print('C.__init__')

若是你运行这段代码就会发现 Base.__init__() 被调用两次,以下所示:spa

>>> c = C()
Base.__init__
A.__init__
Base.__init__
B.__init__
C.__init__
>>>

可能两次调用 Base.__init__() 没什么坏处,但有时候却不是。 另外一方面,假设你在代码中换成使用 super() ,结果就很完美了:代理

class Base:
    def __init__(self):
        print('Base.__init__')

class A(Base):
    def __init__(self):
        super().__init__()
        print('A.__init__')

class B(Base):
    def __init__(self):
        super().__init__()
        print('B.__init__')

class C(A,B):
    def __init__(self):
        super().__init__()  # Only one call to super() here
        print('C.__init__')

运行这个新版本后,你会发现每一个 __init__() 方法只会被调用一次了:code

>>> c = C()
Base.__init__
B.__init__
A.__init__
C.__init__
>>>

为了弄清它的原理,咱们须要花点时间解释下Python是如何实现继承的。 对于你定义的每个类,Python会计算出一个所谓的方法解析顺序(MRO)列表。 这个MRO列表就是一个简单的全部基类的线性顺序表。例如:htm

>>> C.__mro__
(<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>,
<class '__main__.Base'>, <class 'object'>)
>>>

为了实现继承,Python会在MRO列表上从左到右开始查找基类,直到找到第一个匹配这个属性的类为止。

而这个MRO列表的构造是经过一个C3线性化算法来实现的。 咱们不去深究这个算法的数学原理,它实际上就是合并全部父类的MRO列表并遵循以下三条准则:

  • 子类会先于父类被检查
  • 多个父类会根据它们在列表中的顺序被检查
  • 若是对下一个类存在两个合法的选择,选择第一个父类

老实说,你所要知道的就是MRO列表中的类顺序会让你定义的任意类层级关系变得有意义。

当你使用 super() 函数时,Python会在MRO列表上继续搜索下一个类。 只要每一个重定义的方法统一使用 super() 并只调用它一次, 那么控制流最终会遍历完整个MRO列表,每一个方法也只会被调用一次。 这也是为何在第二个例子中你不会调用两次 Base.__init__() 的缘由。

super() 有个使人吃惊的地方是它并不必定去查找某个类在MRO中下一个直接父类, 你甚至能够在一个没有直接父类的类中使用它。例如,考虑以下这个类:

class A:
    def spam(self):
        print('A.spam')
        super().spam()

若是你试着直接使用这个类就会出错:

>>> a = A()
>>> a.spam()
A.spam
Traceback (most recent call last):
    File "<stdin>", line 1, in <module>
    File "<stdin>", line 4, in spam
AttributeError: 'super' object has no attribute 'spam'
>>>

可是,若是你使用多继承的话看看会发生什么:

>>> class B:
...     def spam(self):
...         print('B.spam')
...
>>> class C(A,B):
...     pass
...
>>> c = C()
>>> c.spam()
A.spam
B.spam
>>>

你能够看到在类A中使用 super().spam() 实际上调用的是跟类A毫无关系的类B中的 spam() 方法。 这个用类C的MRO列表就能够彻底解释清楚了:

>>> C.__mro__
(<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>,
<class 'object'>)
>>>

在定义混入类的时候这样使用 super() 是很广泛的。能够参考8.13和8.18小节。

然而,因为 super() 可能会调用不是你想要的方法,你应该遵循一些通用原则。 首先,确保在继承体系中全部相同名字的方法拥有可兼容的参数签名(好比相同的参数个数和参数名称)。 这样能够确保 super() 调用一个非直接父类方法时不会出错。 其次,最好确保最顶层的类提供了这个方法的实现,这样的话在MRO上面的查找链确定能够找到某个肯定的方法。

在Python社区中对于 super() 的使用有时候会引来一些争议。 尽管如此,若是一切顺利的话,你应该在你最新代码中使用它。 Raymond Hettinger为此写了一篇很是好的文章 “Python’s super() Considered Super!” , 经过大量的例子向咱们解释了为何 super() 是极好的。

参考地址:https://python3-cookbook.readthedocs.io/zh_CN/latest/c08/p07_calling_method_on_parent_class.html
相关文章
相关标签/搜索