鱼C论坛

 找回密码
 立即注册
查看: 1556|回复: 5

[已解决]第38课关于super功能

[复制链接]
发表于 2017-7-19 10:34:51 | 显示全部楼层 |阅读模式

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

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

x
没太搞明白super是如何继续父类的
第六题的代码

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…")

两个问题
1、为何最终能实现 进入D,进入B,进入C,进入A,离开A,离开C,离开B,离开D这个逻辑的
在D中是按什么顺序继续B,C父类的功能的
2、把C中的super给注释掉,执行D的时候,就不会进入A,离开A了

没搞懂逻辑,求指导
最佳答案
2017-7-19 13:58:11
2017-07-19_135644.png
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2017-7-19 10:57:31 | 显示全部楼层
本来的继承结构是这样的
搜狗截图20170719105122.jpg
但如果执行过程中也按照这种方式初始化类,就会导致A类被初始化两次,因为B、C类中都有super().__init__(),这样是不符合最初的目的的,A类应该只被初始化一次才对

为了解决这个问题引入了C3算法,在继承的时候从左到右广度优先,所以实际的继承树变成了这样
搜狗截图20170719105127.jpg
也就是说,实际过程中B的父类变成了C,这样就实现了【进入D,进入B,进入C,进入A,离开A,离开C,离开B,离开D】这个逻辑
当你把C中的super给注释掉,那么C就不在去找他的父类A,也就没有了【进入A,离开A】这一步

再进一步,如果你把B中的super().__init__()注释掉,那么C类都不会初始化,就变成了【进入D,进入B, 离开B,离开D】

评分

参与人数 1荣誉 +5 鱼币 +5 贡献 +2 收起 理由
mymmumu + 5 + 5 + 2 谢谢,没时间自己去查原理,你的解释一下子.

查看全部评分

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

使用道具 举报

 楼主| 发表于 2017-7-19 11:54:29 | 显示全部楼层
shuofxz 发表于 2017-7-19 10:57
本来的继承结构是这样的

但如果执行过程中也按照这种方式初始化类,就会导致A类被初始化两次,因为B、C ...

class A():
    def __init__(self):
        print("进入A…")
        print("离开A…")


class Test():
    def __init__(self):
        print('开始test')
        print('结束test')

class B(A):
    def __init__(self):
        print("进入B…")
        
        super().__init__()
        
        print("离开B…")
        
class C(Test):
    def __init__(self):
        print("进入C…")
        
        super().__init__()
        
        print("离开C…")

class D(B,C):
    def __init__(self):
        print("进入D…")
        super().__init__()
        print("离开D…")

类似这种情况,B和C的父类不同,看了下执行结果,D就只会执行B及其父类了,不 会执行后续C类中的动作
如果有类似需求,只能在D的代码中,输入D.__init__么?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2017-7-19 11:55:36 | 显示全部楼层
class D(B,C):
    def __init__(self):
        print("进入D…")
        super().__init__()
        C.__init__(self)
        print("离开D…")
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2017-7-19 12:28:05 | 显示全部楼层
mymmumu 发表于 2017-7-19 11:54
class A():
    def __init__(self):
        print("进入A…")


采用super()方式时,会自动找到第一个多继承中的第一个父类
也就是只找一条主干

要想找其他主干需要用 类名.__init__()  执行


可以看看这篇文章
http://www.jackyshen.com/2015/08/19/multi-inheritance-with-super-in-Python/

评分

参与人数 1荣誉 +3 鱼币 +3 贡献 +3 收起 理由
mymmumu + 3 + 3 + 3

查看全部评分

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

使用道具 举报

发表于 2017-7-19 13:58:11 | 显示全部楼层    本楼为最佳答案   
2017-07-19_135644.png
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2024-4-24 12:21

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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