|
|
发表于 2025-12-11 06:27:29
|
显示全部楼层
要理解 “D.__init__需要 arg 参数但未传递”,关键在于多继承的方法解析顺序(MRO) 和super () 的参数传递逻辑,结合你的代码结构一步步拆解就很清晰了:
先明确类的继承关系和每个类的__init__参数
假设已修正构造方法名为__init__(原代码是_init_,这是基础错误),各阶级的__init__定义如下:
python
运行
class A:
def __init__(self): # 无参数
super().__init__()
class B:
def __init__(self): # 无参数
super().__init__()
class C(A):
def __init__(self, arg): # 需要1个参数arg
super().__init__() # 调用时没传参数
class D(B):
def __init__(self, arg): # 需要1个参数arg
super().__init__() # 调用时没传参数
class E(C, D):
def __init__(self, arg): # 需要1个参数arg
super().__init__(arg) # 调用时传了arg
核心:MRO(方法解析顺序)决定调用链
当实例化E时(e = E(520)),Python 会按MRO 顺序调用所有相关类的__init__。对于E(C, D),MRO 顺序是:E → C → A → D → B → object
(可以通过E.__mro__查看这个顺序,这是 Python 多继承中 “先左后右、深度优先” 的解析规则)
一步步看参数传递过程
调用 E 的__init__:E.__init__(self, 520) 执行,内部调用 super().__init__(520)。根据 MRO,super()此时指向C,所以实际调用 C.__init__(self, 520)(参数520被传递给 C,符合 C 需要arg的要求)。
调用 C 的__init__:C.__init__(self, 520) 执行,内部调用 super().__init__()(注意:这里没传任何参数)。根据 MRO,super()此时指向A,所以实际调用 A.__init__(self)(A 不需要参数,没问题)。
调用 A 的__init__:A.__init__(self) 执行,内部调用 super().__init__()(依然没传参数)。根据 MRO,super()此时指向D(因为 A 的下一个是 D),所以实际调用 D.__init__(self)。
问题出在这里:调用 D 的__init__D.__init__ 的定义是 def __init__(self, arg):,它必须接收 1 个参数 arg,但此时从 A 的super()过来的调用是 D.__init__(self)(没有传arg)。因此,Python 会报错:TypeError: D.__init__() missing 1 required positional argument: 'arg'。
一句话总结
因为 MRO 顺序要求在调用完 A 的__init__后必须调用 D 的__init__,但 D 的__init__需要arg参数,而整个调用链中(从 C 到 A 的super())没有传递这个参数,导致 D 的__init__参数缺失。 |
|