super() 按 mro 查找下一个类的同名方法,返回代理对象实现向上委托;它避免硬编码父类名、确保多重继承中初始化只执行一次,并适用于各类方法和描述符,但须在类方法中调用且参数匹配。

super() 不是简单地调用父类方法,而是按 MRO(方法解析顺序)查找下一个类中的同名方法。它返回一个代理对象,用于在继承链中“向上委托”,避免硬编码父类名,也解决多重继承时的重复调用问题。
super 在单继承中简化父类调用
传统写法 ParentClass.method(self) 会把父类名写死,一旦父类改名或重构就容易出错。用 super().method() 更灵活、可维护性更强:
- 子类重写 __init__ 时,用 super().__init__() 确保父类初始化逻辑执行
- 重写普通方法(如 display())后仍需复用父类逻辑,可用 super().display()
- 不依赖具体父类名,支持类名变更、混入类(Mixin)等动态场景
super 是多重继承正确协作的关键
在钻石继承结构(A ← B, C ← D)中,多个父类都调用 super().__init__(),Python 会按 MRO 顺序逐个触发,保证每个类的 __init__ 只执行一次:
- MRO 可通过 ClassName.__mro__ 查看,例如 D.__mro__ 返回元组 (D, B, C, A, object)
- 所有参与类必须统一使用 super(),否则调用链断裂,部分初始化被跳过
- 典型反例:B 类用 A.__init__(self),C 类用 super().__init__() → A 的 __init__ 被执行两次或漏执行
super 在描述符、属性和特殊方法中保持一致性
不仅限于 __init__,任何需要沿继承链向上查找的方法都适用:
立即学习“Python免费学习笔记(深入)”;
- 重写 __str__ 或 __eq__ 时,用 super().__str__() 复用基础行为
- 自定义描述符(如 __get__)中,通过 super().__get__() 委托给上级描述符逻辑
- 动态类构建(如元类、__new__)中,super().__new__() 是安全创建实例的标准方式
常见误区与注意事项
super 的行为高度依赖调用位置和参数匹配,容易出错:
- super() 在类外部(如模块顶层)会报 RuntimeError,它只能在类方法中使用
- 参数不匹配:若父类方法要求参数,而 super().method() 没传全,会抛 TypeError
- 误以为 super() 总是调用直接父类——实际取决于 MRO,可能跳过中间类(尤其在 Mixin 场景)
- 在静态方法(@staticmethod)中不能用 super(),因无隐式 self 或 cls










