LEEFEI571 发表于 2025-4-6 14:28:29

钻石继承MRO的困惑

>>> class A:
...   def __init__(self):
...         print("哈喽,我是A~")
...
>>> class B1(A):
...   def __init__(self):
...      super().__init__()
...      print("哈喽,我是B1~")
...               
>>> class B2(A):
...   def __init__(self):
...         super().__init__()
...         print("哈喽,我是B2~")
...
>>> class C(B1, B2):
...   def __init__(self):
...         super().__init__()
...         print("哈喽,我是C~")
...
>>> c = C()
执行的结果是
哈喽,我是A~
哈喽,我是B2~
哈喽,我是B1~
哈喽,我是C~

我的问题是类C首先继承的是类B1,然后类B1继承类A,然后类C再继承类B2,类B2再继承类C.按照从左到右的原则,我理解的执行结果应该如下:
哈喽,我是A~
哈喽,我是B1~
哈喽,我是B2~
哈喽,我是C~
为什么执行结果却是先执行类B2里的语句,然后再去执行类B1里的语句?

FishC 发表于 2025-4-6 14:29:22

关于Python钻石继承MRO的执行顺序解析

这个问题与Python的C3线性化算法(MRO计算规则)有关,你的困惑非常典型!让我们分步解释:


[*]Python的继承顺序不是简单的深度优先,而是通过C3算法合并所有父类的MRO
[*]查看类C的MRO顺序会更清晰,可以用 print(C.__mro__) 查看:

(<class '__main__.C'>, <class '__main__.B1'>, <class '__main__.B2'>, <class '__main__.A'>, <class 'object'>)

[*]关键点:super()的调用是沿着MRO链向后传递的


执行过程解析
1. 当调用C()时,C的__init__中的super()指向B1
2. 进入B1的__init__后,B1的super()指向B2(根据MRO顺序)
3. 进入B2的__init__后,B2的super()指向A
4. A执行完毕后,按MRO反方向回溯完成剩余调用


[*]A.__init__ → 打印"A"
[*]回到B2.__init__ → 打印"B2"
[*]回到B1.__init__ → 打印"B1"
[*]最后回到C.__init__ → 打印"C"


为什么不是B1→B2?
因为MRO的合并规则要求:
1. 子类永远在父类前
2. 多个父类保持声明顺序
3. 继承图中不能有循环

在你的案例中,最终的MRO顺序是:
C → B1 → B2 → A → object
(可通过C3算法公式验证)

这个设计避免了传统多重继承的"菱形问题",建议通过 类名.__mro__ 随时验证继承顺序。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
页: [1]
查看完整版本: 钻石继承MRO的困惑