鱼C论坛

 找回密码
 立即注册
查看: 83|回复: 1

钻石继承MRO的困惑

[复制链接]
发表于 2025-4-6 14:28:29 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
>>> 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里的语句?
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2025-4-6 14:29:22 | 显示全部楼层
关于Python钻石继承MRO的执行顺序解析

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

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

    1. (<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 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-5-5 20:14

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表