鱼C论坛

 找回密码
 立即注册
查看: 1563|回复: 2

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

[复制链接]
发表于 2020-7-14 22:20:26 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
对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…
最佳答案
2020-7-14 22:24:21
本帖最后由 sunrise085 于 2020-7-14 22:28 编辑

super调用按照__mro__列表顺序调用,你的这个继承中,__mro__列表是:<class '__main__.D'>,  <class'__main__.B'>, <class '__main__.C'>, <class '__main__.A'>,<class 'object'>
可以通过 D.__mro__ 查看该列表
我写过一个总结帖
Python细节之6、类的多继承中super函数的调用顺序

你这是菱形继承的问题,小甲鱼也特意写过一个帖子:多重继承的陷阱:钻石继承(菱形继承)问题
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2020-7-14 22:24:21 | 显示全部楼层    本楼为最佳答案   
本帖最后由 sunrise085 于 2020-7-14 22:28 编辑

super调用按照__mro__列表顺序调用,你的这个继承中,__mro__列表是:<class '__main__.D'>,  <class'__main__.B'>, <class '__main__.C'>, <class '__main__.A'>,<class 'object'>
可以通过 D.__mro__ 查看该列表
我写过一个总结帖
Python细节之6、类的多继承中super函数的调用顺序

你这是菱形继承的问题,小甲鱼也特意写过一个帖子:多重继承的陷阱:钻石继承(菱形继承)问题
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-7-14 22:27:34 | 显示全部楼层


MRO 的顺序基本就是:在避免同一类被调用多次的前提下,使用广度优先和从左到右的原则去寻找需要的属性和方法。

在继承体系中,C3 算法确保同一个类只会被搜寻一次。例子中,如果一个属性或方法在 D 类中没有被找到,Python 就会搜寻 B 类,然后搜索 C类,如果都没有找到,会继续搜索 B 的基类 A,如果还是没有找到,则抛出“AttributeError”异常。

扩展阅读:
多重继承的陷阱:钻石继承(菱形继承)问题


用 super 调用顺序 是按照 MRO 顺序和遵循 C3算法 来的 ,参考扩展阅读的材料吧


想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-20 01:56

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表