python的面向对象编程(oop)中,继承是一个强大的工具,但它也常常是初学者(甚至有经验的开发者)的困惑之源。核心的困惑点通常围绕:
为什么子类的
__init__必须手动调用父类的__init__?super()到底是什么,它和直接用父类名调用有何区别?
本文将通过“造房子”的类比和“钻石问题”的深度解析,彻底澄清这些概念。
第1部分:核心问题——__init__的“覆盖”机制
要理解为何需要 super(),首先必须理解python的方法搜索规则。
当在子类实例上调用一个方法时(如 __init__),python的搜索顺序是:
先在子类中寻找该方法。
如果找到了,立即停止搜索并执行。
如果没找到,才去父类中寻找。
这个规则会导致一个常见问题:子类 __init__ 会覆盖父类 __init__。
类比:造房子(standardhousevsdeluxehouse)
假设我们有一个“标准房子”的设计图(父类):
class standardhouse:
def __init__(self, foundation_type):
# 建造房子的第一步:打地基
self.foundation = foundation_type
print(f"地基已打好,类型是: {self.foundation}")
def get_info(self):
print(f"这是一座标准房子,地基是 {self.foundation}")
现在,我们想设计一个“豪华房子”(子类),它在“标准房子”的基础上,还想额外添加一个“游泳池”。为此,我们必须定义它自己的 __init__:
错误的继承(“覆盖”导致的问题)
class deluxehouse(standardhouse):
# 子类定义了自己的 __init__
def __init__(self, pool_size):
# 建造房子的第一步:建游泳池
self.pool = pool_size
print(f"游泳池已建好,大小是: {self.pool} 平米")
# --- 尝试建造 ---
print("--- 准备建造豪华房子 ---")
my_deluxe_house = deluxehouse(50)
# --- 尝试获取信息 ---
try:
my_deluxe_house.get_info() # 调用继承来的方法
except attributeerror as e:
print(f"\n!!! 出错了: {e}")
执行结果:
--- 准备建造豪华房子 ---
游泳池已建好,大小是: 50 平米
!!! 出错了: 'deluxehouse' object has no attribute 'foundation'
错误分析:
创建
deluxehouse(50)时,python 在deluxehouse中找到了__init__。它立刻停止搜索,并执行了子类的
__init__。父类 standardhouse 的 __init__ 从未被调用。
因此,
self.foundation(地基)这个属性根本没有被创建。当
get_info()试图访问self.foundation时,程序崩溃。
第2部分:解决方案——super()与self的真相
deluxehouse 的建造者在建游泳池之前,必须显式地(手动地)先去执行“标准房子”的建造流程(打地基)。super() 就是这个“手动调用父类”的命令。
1.super()的含义
super():一个特殊的函数,返回一个“父类”的代理对象。super().__init__(...):通过该代理对象,调用父类的__init__方法。
2.self的真相
从始至终,只有一个 self 对象。self 就是那个“正在被建造的实例”。
standardhouse的建造队(父类)和deluxehouse的建造队(子类)都在同一栋房子(self)上施工。super()就是子类团队用来“呼叫”父类团队的电话。
正确的代码:
class deluxehouse(standardhouse):
# 子类 __init__ 必须接收 *所有* 参数(父类的 + 自己的)
def __init__(self, foundation_type, pool_size):
print("--- [子类] deluxehouse.__init__ 开始 ---")
# 关键步骤:呼叫父类团队,先把地基打好
# 把属于父类的参数 (foundation_type) 传递过去
super().__init__(foundation_type)
print("--- [子类] 父类工作已完成,现在我来添加新功能 ---")
# 2. 子类现在可以安全地设置自己的属性
self.pool = pool_size
print(f" [子类] 游泳池已建好: {self.pool}")
print("--- [子类] deluxehouse.__init__ 结束 ---")
# --- 尝试建造 ---
my_deluxe_house = deluxehouse("钢筋混凝土", 50)
my_deluxe_house.get_info()
执行流程分析:
调用
deluxehouse(...),进入deluxehouse.__init__。执行
super().__init__(...),暂停子类__init__,跳转到父类standardhouse.__init__。父类
__init__在同一个 self 上设置了self.foundation。父类
__init__结束,返回到子类__init__。子类
__init__继续执行,在同一个 self 上设置self.pool。self对象现在同时拥有了.foundation和.pool。
第3部分:super()vsparentclass.__init__(新旧对比)
既然必须手动调用父类,有两种方法可以做到:
现代方式 (推荐):
super().__init__(...)老式方式 (不推荐):
parentclass.__init__(self, ...)
在简单的“单继承”场景下,两者似乎都能工作:
class vehicle:
def __init__(self, brand):
print(f"[vehicle] 设置品牌为: {brand}")
self.brand = brand
# 1. 现代方式 (super)
class car_modern(vehicle):
def __init__(self, brand, model):
# 语法:
# 1. 不需要写父类名
# 2. 不需要传递 self
super().__init__(brand)
self.model = model
# 2. 老式方式 (父类名)
class car_old(vehicle):
def __init__(self, brand, model):
# 语法:
# 1. 必须 明确写出父类名 (vehicle)
# 2. 必须 手动传递 self
vehicle.__init__(self, brand)
self.model = model
为什么 super() 完胜?
| 特性 | super().__init__(...) (现代) | parentclass.__init__(self, ...) (老式) |
| self 参数 | 隐式传递。python自动处理,代码更简洁。 | 必须手动传递。容易忘记,导致错误。 |
| 可维护性 | 高 (灵活) | 低 (死板 / 易碎) |
| 原因 | 如果父类改名 (如 vehicle -> basevehicle),子类代码无需任何改动。 | 如果父类改名,你必须在所有子类中搜索并替换 vehicle.__init__。 |
| 多重继承 | 唯一正确的方案。 | 完全失效或导致灾难性bug。 |
第4部分:决定性因素——多重继承与“钻石问题”
super() 存在的真正理由是:它能智能地处理复杂的多重继承,而“父类名”调用则会彻底失败。
经典的“钻石问题”:d 继承自 b 和 c,而 b 和 c 都继承自 a。
a
/ \
b c
\ /
d
1.super()的正确演示
super() 会智能地遵循“方法解析顺序”(mro),确保继承链中的每个 __init__ 都被调用一次,且仅被调用一次。
class a:
def __init__(self):
print("a 初始化 (只应被调用一次)")
class b(a):
def __init__(self):
print("b 初始化开始...")
super().__init__()
print("b 初始化结束。")
class c(a):
def __init__(self):
print("c 初始化开始...")
super().__init__()
print("c 初始化结束。")
class d_super(b, c): # 孙子 (使用 super)
def __init__(self):
print("d_super 初始化开始...")
super().__init__()
print("d_super 初始化结束。")
print("--- 测试 super() 的多重继承 ---")
d1 = d_super()
super() 的正确执行结果 (a只调用一次):
--- 测试 super() 的多重继承 ---
d_super 初始化开始...
b 初始化开始...
c 初始化开始...
a 初始化 (只应被调用一次)
c 初始化结束。
b 初始化结束。
d_super 初始化结束。
2. “老式”方法的错误演示
如果 b 和 c 使用“父类名”的方式调用 a:
class b_old(a):
def __init__(self):
print("b_old 初始化开始...")
a.__init__(self) # 明确调用 a
print("b_old 初始化结束。")
class c_old(a):
def __init__(self):
print("c_old 初始化开始...")
a.__init__(self) # 明确调用 a
print("c_old 初始化结束。")
class d_old(b_old, c_old):
def __init__(self):
print("d_old 初始化开始...")
b_old.__init__(self) # 明确调用 b
c_old.__init__(self) # 明确调用 c
print("d_old 初始化结束。")
print("\n--- 测试 '父类名' 的多重继承 (错误演示) ---")
d2 = d_old()
“老式”方法的错误执行结果 (a被调用两次!):
--- 测试 '父类名' 的多重继承 (错误演示) ---
d_old 初始化开始...
b_old 初始化开始...
a 初始化 (只应被调用一次) <-- a 被调用了第1次
b_old 初始化结束。
c_old 初始化开始...
a 初始化 (只应被调用一次) <-- a 被调用了第2次 (!! bug !!)
c_old 初始化结束。
d_old 初始化结束。
分析:a (顶层父类) 的 __init__ 被调用了两次!如果 a 的 __init__ 是在打开一个文件或建立一个数据库连接,这种重复调用很可能会导致资源冲突或数据损坏。
结论与建议
| 对比项 | super() (推荐) | parentclass.__init__(self, ...) (不推荐) |
| 简单继承 | 可用 | 可用(但不推荐) |
| 多重继承 | 完美工作 | 导致bug(如重复调用) |
| 可维护性 | 高 (父类改名不影响子类) | 低 (父类改名必须重构所有子类) |
| 语法 | 简洁 (自动处理self) | 繁琐 (必须手动传递self) |
| 适用范围 | 现代python (3.x) 的标准 | 历史遗留 (python 2 旧代码) |
最终建议:
始终使用 super()。 它更简洁、更灵活,是唯一能正确处理复杂继承(多重继承)的工具。parentclass.__init__ 是一种需要被理解(以便能看懂旧代码)但不再需要被使用的历史写法。
到此这篇关于理解python继承之从__init__覆盖到super()的妙用方法的文章就介绍到这了,更多相关python继承__init__覆盖super()内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
发表评论