python3 super().__init__(),python3的super().__init__()

父类不会自动调用__init__方法

class A:

def __init__(self):

A = 'A'

self.a = 'a'

print('init A')

class B(A):

def __init__(self):

self.b = 'b'

print('init B')

b = B()

print(b.A)

print(b.a)

init B

A

---------------------------------------------------------------------------

AttributeError Traceback (most recent call last)

in

1 b = B()

2 print(b.A)

----> 3 print(b.a)

AttributeError: 'B' object has no attribute 'a'

子类继承父类,子类实例化只会调用自己的构造方法,父类的构造方法不会调用,或者说子类重写了父类的构造方法,所以父类在构造方法中定义的默认属性也无法继承。

实例变量不能继承不是Python继承有问题,上面在class A中定义了一个类变量A就被成功继承下来了,python类中的所有最外层类变量,类方法,静态方法,实例方法该继承都会继承,不过实例变量是定义在__init__构造方法中的,虽然__init__会在实例化时自动执行(不管是父类还是子类),但子类重写了父类方法,所以父类构造没有执行,只是执行了子类构造。不然看下面一个例子:

class E(A):

pass

e = E()

print(e.a)

init A

a

如果子类没有重写父类方法那么父类构造还是自动执行了。所以总的来说父类构造没有执行的原因是**子类重写了父类的__init__方法**

使用super().__init__()手动执行父类的构造方法

如果要手动执行, 首先绝不能这样:

class B(A):

def __init__(self):

self.__init__()

self.b = 'b'

print('init B')

b = B()

print(b.A)

print(b.a)

---------------------------------------------------------------------------

RecursionError Traceback (most recent call last)

in

----> 1 b = B()

2 print(b.A)

3 print(b.a)

in __init__(self)

7 class B(A):

8 def __init__(self):

----> 9 self.__init__()

10 self.b = 'b'

11 print('init B')

... last 1 frames repeated, from the frame below ...

in __init__(self)

7 class B(A):

8 def __init__(self):

----> 9 self.__init__()

10 self.b = 'b'

11 print('init B')

RecursionError: maximum recursion depth exceeded

这样做的后果是会无限递归调用子类自己的__init__

不过可以手动指定父类类名进行调用,比如下面这样:

class B(A):

def __init__(self):

A.__init__(self)

self.b = 'b'

print('init B')

b = B()

print(b.A)

print(b.a)

init A

init B

A

a

这样做是可以成功的,缺点就是太累了,在多继承环境下我们需要指定每一个父类进行调用,比如下面这样

class B(A):

def __init__(self):

A.__init__(self)

self.b = 'b'

print('init B')

class C(A):

def __init__(self):

self.c = 'c'

print('init C')

class D(B,C):

def __init__(self):

B.__init__(self)

C.__init__(self)

self.d = 'd'

print('init D')

d = D()

print(d.b)

print(d.c)

init A

init B

init C

init D

b

c

显然B需要调A,D需要调用B和C,显然不符合python的优雅风格

super().__init__的好处就是统一进行所有父类构造方法的调用,调用的先后根据类型的mro顺序决定,将自动调用所有父类的构造方法

我们这里只关注直系父类,因为对于祖父级别的类构造方法是否调用取决于父类中是否重写了祖父的构造方法,只要在继承的父类中有一个父类重写了祖父的构造方法,那么祖父的构造方法也不会被调用。

我们可以根据.mro()查看类型的MRO表

print(D.mro())

[, , , , ]

super().__init__()是python3中简化后的方法,功能和super(self.__class__, self).__init__()一样

承接Matlab、Python和C++的编程,机器学习、计算机视觉的理论实现及辅导,本科和硕士的均可,咸鱼交易,专业回答请走知乎,详谈请联系QQ号757160542,非诚勿扰。

本文同步分享在 博客“于小勇”(CSDN)。

如有侵权,请联系 support@oschina.cn 删除。

本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值