super 使用以及原理

时间:2022-11-12 17:06:10

用super也很久了,但是一直没有关注过他的原理。最近开始越来越多关注python更底层的实现和奇技淫巧。看到该方法越发使用得多所以也研究了一波

 

平时单继承可能是我们遇到最多的情况。无非就是类似情况。

class A(object):
    def __init__(self, a, b):
        print 'times gone away %s, %s' % (a, b)


class B(A):
    def __init__(self, a, b):
        super(B, self).__init__(a, b)

B(1, 2)
times gone away 1, 2

这个例子的类B继承了类A,然后在初始化方法里面调用了父类A的初始化方法并且传入了参数。

其实我想说就这个 例子来看。。这样也可以

class A(object):
    def __init__(self, a, b):
        print 'times gone away %s, %s' % (a, b)


class B(A):
    pass


B(1, 2)

当然。调用父类初始化方法之后,我们依然在后面可以做一些 其他的操作,但是第二个例子就做不到了注意。

 

super看上去 也可以通过直接写父类A.__init__(self, a, b)来实现但是,当我们遇到多继承的时候情况就有所改变。super可以帮忙利用方法解析顺序(Method Resolution Order, MRO)列表来寻找按顺序顺着父类。在多继承的情况下,初始化循序并非我们所想的自底向上。

而是顺着mro列表从左到右。

B.mro()   # or C.__mro__ or C().__class__.mro()
[<class '__main__.B'>, <class '__main__.A'>, <type 'object'>]

如上所示有三种方法查看该类的mro顺序。

从上面返回的数组我们可以看到,从顺序上看是B到A到 object的继承关系。

而super方法调用的本质是:

def super(cls, inst):
    mro = inst.__class__.mro()
    return mro[mro.index(cls) + 1]

我们将类名cls 和实例self传入,然后由mro找到自己的类位置并且+1 返回下个继承类调用位置。 而不是自底向上的寻找父类。 有这个公式可能就更容易明白了。

preference的这篇文章举了一个 完整的多继承例子,但是我觉得 只要你理解了 python中寻找父类的方法不是自底向上 而是依赖mro算法的话,这样就不难理解了。

 

Reference:

https://segmentfault.com/a/1190000007426467  Python: 你不知道的 super