心空妙有 发表于 2020-7-14 22:20:26

对python在子类中通过“super”方法调用父类的过程的疑惑

对python在子类中通过“super”方法调用父类的过程的疑惑

为什么像第二种,就没有进入C,离开C了;第三种没有进入,离开A了

第一种:
class A():
    def __init__(self):
      print("进入A…")
      print("离开A…")

class B(A):
    def __init__(self):
      print("进入B…")
      super().__init__()
      print("离开B…")

class C(A):
    def __init__(self):
      print("进入C…")
      super().__init__()
      print("离开C…")

class D(B, C):
    def __init__(self):
      print("进入D…")
      super().__init__()
      print("离开D…")

d = D()

第一种结果:
进入D…
进入B…
进入C…
进入A…
离开A…
离开C…
离开B…
离开D…

第二种:
class A():
    def __init__(self):
      print("进入A…")
      print("离开A…")

class B(A):
    def __init__(self):
      print("进入B…")
      print("离开B…")

class C(A):
    def __init__(self):
      print("进入C…")
      super().__init__()
      print("离开C…")

class D(B, C):
    def __init__(self):
      print("进入D…")
      super().__init__()
      print("离开D…")

d = D()

第二种结果:
进入D…
进入B…
离开B…
离开D…

第三种:
class A():
    def __init__(self):
      print("进入A…")
      print("离开A…")

class B(A):
    def __init__(self):
      print("进入B…")
      super().__init__()
      print("离开B…")

class C(A):
    def __init__(self):
      print("进入C…")
      print("离开C…")

class D(B, C):
    def __init__(self):
      print("进入D…")
      super().__init__()
      print("离开D…")

d = D()

第三种结果:
进入D…
进入B…
进入C…
离开C…
离开B…
离开D…

sunrise085 发表于 2020-7-14 22:24:21

本帖最后由 sunrise085 于 2020-7-14 22:28 编辑

super调用按照__mro__列表顺序调用,你的这个继承中,__mro__列表是:<class '__main__.D'>,<class'__main__.B'>, <class '__main__.C'>, <class '__main__.A'>,<class 'object'>
可以通过 D.__mro__ 查看该列表
我写过一个总结帖
Python细节之6、类的多继承中super函数的调用顺序

你这是菱形继承的问题,小甲鱼也特意写过一个帖子:多重继承的陷阱:钻石继承(菱形继承)问题

Twilight6 发表于 2020-7-14 22:27:34



MRO 的顺序基本就是:在避免同一类被调用多次的前提下,使用广度优先和从左到右的原则去寻找需要的属性和方法。

在继承体系中,C3 算法确保同一个类只会被搜寻一次。例子中,如果一个属性或方法在 D 类中没有被找到,Python 就会搜寻 B 类,然后搜索 C类,如果都没有找到,会继续搜索 B 的基类 A,如果还是没有找到,则抛出“AttributeError”异常。

扩展阅读:
多重继承的陷阱:钻石继承(菱形继承)问题

用 super 调用顺序 是按照 MRO 顺序和遵循 C3算法 来的 ,参考扩展阅读的材料吧


页: [1]
查看完整版本: 对python在子类中通过“super”方法调用父类的过程的疑惑