python类与对象详解(2):子类调用父类方法的两种方式

2016-04-16  本文已影响1115人  KillerManA

当子类继承父类的一些方法并且子类已经覆盖此方法时,我们如何调用父类中的函数?这里提供两种调用方式:

class Base:
    def __init__(self):
        print("Base.__init__")
class A(Base):
    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__")

运行一下代码:

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

这里可以清晰的看到当我们进行多重继承的时候父类被多次调用,看似没有任何问题,当时当代码复杂到一定程度时会摸不着头脑,产生的错误莫名其妙。我们再使用supper进行相同的操作:

class Base:
    def __init__(self):
        print("Base.__init__")
class A(Base):
    def __init__(self):
        supper().__init__()
        print("A.__init__")
class B(Base):
    def __init__(self):
        supper().__init__()
        print("B.__init__")
class C(A, B):
    def __init__(self):
        supper().__init__()
        print("C.__init__")

运行代码:

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

从得到的结果跟上面的代码运行结果进行对比可以得出以下结果:
当我们使用supper进行父类方法调用时,只会调用一次父类的方法

我们再来看看这个调用是怎样实现的:
python进行继承时,针对每个定义的类,python都会计算出一个方法解析顺序列表(MRO)MRO只是简单地针对所有的基类进行现行排列。实现继承时,python会从MRO列表中最左边的类开始,从左到右进行查找,直到找到待查的属性时为止。这是python继承的机制,那么MRO又是怎样实现的?
简单来说这是针对父类的一种归并排序,需要满足3个约束:

再来回顾刚才的例子:
MRO会控制最终遍历整个MRO列表,并且每个方法只会被调用一次。这里就是刚才我们使用supper时为何只调用一次父类方法的原因。

上一篇下一篇

猜你喜欢

热点阅读