当子类继承父类的一些方法并且子类已经覆盖此方法时,我们如何调用父类中的函数?这里提供两种调用方式:
- 通过在子类中实例化父类的方式调用父类函数
class Base:
def __init__(self):
print("Base.__init__")
class A(Base):
Base.__init__(self)
print("A.__init__")
这种方法比较奔放,对于大多数还是正常的,但是对于多重继承的时候会重复调用父类。解决办法就是使用第二种方法进行调用。
- 通过supper调用父类函数
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会控制最终遍历整个MRO列表,并且每个方法只会被调用一次。这里就是刚才我们使用supper时为何只调用一次父类方法的原因。