多继承指的是子类继承多个父类,可以通过三种方式访问父类的方法:
多继承中的菱形问题代码:
# 菱形问题,s1和s2都继承自P然后Gs继承自S1和S2形成
class P(object):
def __init__(self):
self.name = None
print("p")
class S1(P):
def __init__(self):
self.name = None
print("s1")
P.__init__(self)
class S2(P):
def __init__(self):
self.name = None
print("s2")
P.__init__(self)
class Gs(S1, S2):
def __init__(self):
self.name = None
print("G2")
S1.__init__(self)
S2.__init__(self)
if __name__ == '__main__':
g = Gs()
运行结果为:
G2
s1
p
s2
p # P类的init方法被执行了两遍
class P(object):
def __init__(self):
self.name = None
print("p")
class S1(P):
def __init__(self):
self.name = None
print("s1")
super().__init__() # 使用super().__init__()时不用传参数self
class S2(P):
def __init__(self):
self.name = None
print("s2")
super().__init__()
class Gs(S1, S2):
def __init__(self):
self.name = None
print("G2")
super().__init__()
if __name__ == '__main__':
g = Gs()
输出结果:
G2
s1
s2
p # P类的init方法只执行了一边
可以通过类名.__MRO__属性查找出来当前类的调用顺序,其顺序由C3算法来决定,保证每一个类只调用一次。
使用__mro__打印继承的顺序
print(Gs.__mro__) # 输出结果如下:
(<class '__main__.Gs'>, <class '__main__.S1'>, <class '__main__.S2'>, <class '__main__.P'>, <class 'object'>)
调用继承方法时会按照此顺序进行调用,顺序是Python自己计算出来的,C3算法,我们不需要了解其中是怎么算的.
如果不想按照这样的顺序进行调用,可以使用super(指定某个类名, self).父类方法()
从指定类名的MRO下一级开始调用,如将Gs类中的super方法改成这样的super(S2,self).__init__()
此时在执行常见Gs类的实例对象的时候的输出结果为G2 P
,此时就可以理解什么是从指定类名的下一级开始调用了
单继承用哪种方式调用父类方法都可以,基本上无差别,但是建议super()的方式。
super()单继承只需要传父类参数,但是多继承必须传全部参数,可以使用多值参数。
class Parent(object):
def __init__(self, name, *args, **kwargs): # 为避免多继承报错,使用不定长参数,接受参数
print('parent的init开始被调用')
self.name = name
print('parent的init结束被调用')
class Son1(Parent):
def __init__(self, name, age, *args, **kwargs): # 此处接收到Grandson传来的name和age参数,其余的参数由*args存储继续向上传递
print('Son1的init开始被调用')
self.age = age
super().__init__(name, *args, **kwargs) # 为避免多继承报错,使用不定长参数,接受参数
print('Son1的init结束被调用')
class Son2(Parent):
def __init__(self, name, gender, *args, **kwargs): # 为避免多继承报错,使用不定长参数,接受参数
print('Son2的init开始被调用')
self.gender = gender
super().__init__(name, *args, **kwargs) # 为避免多继承报错,使用不定长参数,接受参数
print('Son2的init结束被调用')
class Grandson(Son1, Son2):
def __init__(self, name, age, gender): # 此处即使Grandson类不需要任何参数但是其父类有需要的参数,要在此处把继承的所有父类的和祖父类的所有参数在此处传递
print('Grandson的init开始被调用')
super().__init__(name, age, gender) # 调用父类的init方法,把参数全部传递
print('Grandson的init结束被调用')
print(Grandson.__mro__) # 打印此多继承的mro顺序
gs = Grandson('grandson', 12, '男')
print('姓名:', gs.name)
print('年龄:', gs.age)
print('性别:', gs.gender)
super().__init__
相对于类名.__init__
,在单继承上用法基本无差*args
和**wkargs
时子类传参也应该这样使用,带*
号)这也是为何多继承需要全部传参的一个原因