
什么是钻石继承问题
钻石继承(Diamond Inheritance)指在多继承中,两个父类继承自同一个祖父类,而子类又同时继承这两个父类,形成类似菱形的继承结构。Python 中虽然没有 C++ 那样的“虚继承”机制,但由于使用 MRO(Method Resolution Order,方法解析顺序)和 C3 线性化算法,能明确决定调用哪个版本的方法——但若设计不当,仍会导致方法被意外跳过、重复执行或属性覆盖,这就是常说的“钻石继承问题”。
MRO 是关键:看懂 Python 的调用顺序
Python 通过 ClassName.__mro__ 或 help(ClassName) 查看类的 MRO 元组,它决定了属性和方法按什么顺序查找。C3 算法保证:子类优先于父类、各父类按声明顺序从左到右、保持所有祖先类的相对顺序。
- 例如:
class D(B, C): pass,且B和C都继承自A,则D.__mro__通常是(D, B, C, A, object) - 如果
B.__init__和C.__init__都直接调用A.__init__(),而没用super(),A.__init__就会被执行两次 - 正确做法是所有中间类统一用
super().__init__(),让控制权沿 MRO 向下传递
用 super() 协作式调用代替显式父类名
这是解决钻石继承中最核心的实践。协作式调用要求所有参与继承链的类都使用 super(),而不是硬编码父类名(如 A.__init__(self)),从而让每个方法只执行一次,并准确衔接下一个 MRO 类。
- 错误写法:
B.__init__里写A.__init__(self),C.__init__也写A.__init__(self)→A.__init__被调两次 - 正确写法:
B.__init__写super().__init__(),C.__init__同样写super().__init__(),A.__init__只在 MRO 中第一次出现时被触发 - 注意:所有类(包括
A)的__init__也应接受并转发未处理的参数,或使用**kwargs兼容不同路径传入的参数
避免冲突的补充建议
除统一使用 super() 外,还可结合以下方式提升健壮性:
立即学习“Python免费学习笔记(深入)”;
- 尽量减少不必要的多继承;优先考虑组合(composition)替代继承(如用属性引用其他类实例)
- 若必须多继承,把共享逻辑抽离为 mixin 类,并确保其方法都是“可跳过”的(比如检查
hasattr(self, '_initialized')防重入) - 对关键初始化逻辑加标记(如
self._a_initialized = True),在调用前判断是否已执行 - 测试时打印 MRO 并手动模拟调用链,验证每一步是否符合预期










