一、super()
class Person:
def __init__(self,name)
self.name = name
class Male(Person):
def __init__(self,name):
super().__init__(name)
self.gender = "male"
super调用的不是父类中的方法,他调用的是MRO列表里的上一个函数方法
super()
函数可以放入两个参数,第一个参数决定mro链上从哪个class开始找
如上图,从Person后的一个class开始找,那么这个class是Animal,此时只能传入age一个参数
二、MRO:
在python中,一个class如何从它的父类中找应该优先使用那个父类的函数,这个顺序就是MRO(method resolution order)。
MRO会把一个class的所有父类和自己做一个线性化(serialzation),这个class在这个顺序中是最高优先级。
class A:
def say(self):
print("A")
class B(A):
def say(self):
print("B")
class C(A):
pass
class M(C,B):
pass
>>> m = M()
>>> m.say()
结果: B
这里class B 的优先级比A高,我们打印mro看一下
>>> print(M.mro())
[<class '__main__.M'>, <class '__main__.C'>, <class '__main__.B'>, <class '__main__.A'>, <class 'object'>]
如果class B
没有继承A的话,优先级顺序A就会在前面。当我们建立了新的类,或是继承了其他类的时候,python首先会根据继承关系计算MRO。并依次在所计算出的顺序下,依次寻找方法。
MRO算法细节:
python的MRO使用的C3 linearization算法,其之所以被命名为C3是因为其满足3个属性(it is consistent with three pro perties)
- a consistent extended precedence graph
如果两个类不具有直接的继承关系,那么找到两个类的最小公共子类,这个最小公共子类的多继承顺序靠前的分支上的类具有高优先级 - preservation of local precedence order
局部优先顺序,当一个类继承了多个类的时候,会优先使用写在前面的类,不仅如此,当有其他类使用这个继承其他类的类的时候,也会保持这个特性 - fitting a monotonicity criterion
- 单调性,子类不改变父类的方法
- 任何一个class使用的方法必须来自他的直接父类(direct super class)使用的方法