python调用父类方法_对python在子类中通过“super”方法调用父类的过程的疑惑

对python在子类中通过“super”方法调用父类的过程的疑惑

为什么像第二种,就没有进入C,离开C了;第三种没有进入,离开A了

第一种:

class A():

def __init__(self):

print("进入A…")

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(B, C):

def __init__(self):

print("进入D…")

super().__init__()

print("离开D…")

d = D()

第一种结果:

进入D…

进入B…

进入C…

进入A…

离开A…

离开C…

离开B…

离开D…

第二种:

class A():

def __init__(self):

print("进入A…")

print("离开A…")

class B(A):

def __init__(self):

print("进入B…")

print("离开B…")

class C(A):

def __init__(self):

print("进入C…")

super().__init__()

print("离开C…")

class D(B, C):

def __init__(self):

print("进入D…")

super().__init__()

print("离开D…")

d = D()

第二种结果:

进入D…

进入B…

离开B…

离开D…

第三种:

class A():

def __init__(self):

print("进入A…")

print("离开A…")

class B(A):

def __init__(self):

print("进入B…")

super().__init__()

print("离开B…")

class C(A):

def __init__(self):

print("进入C…")

print("离开C…")

class D(B, C):

def __init__(self):

print("进入D…")

super().__init__()

print("离开D…")

d = D()

第三种结果:

进入D…

进入B…

进入C…

离开C…

离开B…

离开D…

回答

了解这种现象需要以下两点:

super() 返回的是一个代理对象,使得子类可以调用父类的方法,这个可以查看官方文档

Python 中存在完整定义的函数查找顺序 (非标准翻译,原文 Method Resolution Order,MRO, 见这个链接)

简单来说,就是调用 super().__ini__() 只是依据MRO查找相应的对象,找到存在目标方法的对象后即调用该对象的方法并**停止本轮查找**,而不会像一眼看过去可能会认为的那样“逐级初始化”父类

MRO可以通过 class property __mro__ 检查

D.__mro__

针对你的例子,第一个例子里,其MRO,即查找顺序为 D, B, C, A (你的3个例子 MRO 是一样的)

每次使用 super() 会使得查找起点从当前往右移动一格,即在 D 内调用 super().__init__() 将从B开始查找,由于B定义了__init__() 方法,本轮查找结束。然而B.__init__()内调用了super().__init__(),使得Python再次依据MRO进行查找,这次从C开始(B 往右一格)查找,由于 C 定义了__init__(), 本轮结束,但是C.__init__() 内部调用了 super().__init__() 再次使得Python基于同样的逻辑调用 A 的__init__(),所以你看到的是

进入D…

进入B…

进入C…

进入A…

离开A…

离开C…

离开B…

离开D…

而不是进入B后进入A这种看上去似乎是“逐级递进”的调用方式

第二个例子中,基于同样的原则,在 D 的 __init__() 中调用 super().__init__(),到B的时候本轮查找结束,由于 B 的__init__() 方法不再创建新的调用链条,调用栈到此结束;第三个例子理由一致,你试试自己理解下第三个例子?

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值