Python3中的super()函数详解

关于Python3中的super()函数

我们都知道,在Python3中子类在继承父类的时候,当子类中的方法与父类中的方法重名时,子类中的方法会覆盖父类中的方法,

那么,如果我们想实现同时调用父类和子类中的同名方法,就需要使用到super()这个函数,用法为super().函数名()

下面是一个例子:

class A1():
    def go(self):
        print("go A1 go")

class A2():
    def go(self):
        print("go A2 go")

class A3():
    def go(self):
        print("go A3 go")

class C(A3):
    pass

class B(A1,A2):
    pass

class D(B,C):
    def go(self):
        print("NMSL")
        super().go()

d1 = D()
d1.go()
print(D.__mro__)

输出结果为:

NMSL
go A1 go
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.A1'>, <class '__main__.A2'>, <class '__main__.C'>, <class '__main__.A3'>, <class 'object'>)

这里的__mro__属性显示了当类调用方法时,如果父类和子类中有同名方法情况下的查找顺序。

如图,当我们实例化D类并调用类中的go方法时,go方法中有一条语句调用了父类的go方法,__mro__属性显示了如何查找这个方法(当然,自身类中的go方法不算),最后我们知道它调用的是A1类中的go方法,那么有些人会疑惑,为什么它不调用更近的的A3类或者A2类中go方法呢,这就要涉及到super()函数实现顺序查找的算法,这个算法即为C3算法。我的另一篇博客中记录了这个算法的原理。

假如D类中本身就没有go方法,那么我们再使用super()函数进行调用go方法,它会选择哪一个父类的呢?

代码如下:

class A1():
    def go(self):
        print("go A1 go")

class A2():
    def go(self):
        print("go A2 go")

class A3():
    def go(self):
        print("go A3 go")

class C(A3):
    pass

class B(A1,A2):
    pass

class D(B,C):
    def gogo(self):
        print("NMSL")
        super().go()

d1 = D()
d1.gogo()
print(D.__mro__)


运行后输出结果:

NMSL
go A1 go
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.A1'>, <class '__main__.A2'>, <class '__main__.C'>, <class '__main__.A3'>, <class 'object'>)

可以看到结果还是这样。

如果D类中不定义方法,仅仅在类的外部通过实例调用go方法,查找过程也是一样的。

如图:

class D(B,C):
pass
d1 = D()
d1.go()
print(D.__mro__)

结果:

NMSL
go A1 go
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.A1'>, <class '__main__.A2'>, <class '__main__.C'>, <class '__main__.A3'>, <class 'object'>)

以上就是对于super()函数的个人看法。

猜你喜欢

转载自www.cnblogs.com/LegendsNeverDie/p/10367473.html