父类不会自动调用__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)
<ipython-input-62-6b0a00f52204> in <module>
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)
<ipython-input-68-6b0a00f52204> in <module>
----> 1 b = B()
2 print(b.A)
3 print(b.a) <ipython-input-67-7b74bc3d523e> 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 ... <ipython-input-67-7b74bc3d523e> 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顺序决定,将自动调用所有父类的构造方法

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

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

print(D.mro())

[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]

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

python3 super().__init__()的更多相关文章

  1. python3 super().__init__() 和 __init__() 的区别

    1.单继承 super().__int__()和 Base.__init__(self)是一样的, super()避免了基类的显式调用. class Base(object): def __init_ ...

  2. 推荐使用的派生方法:super().__init__()

    """ 推荐使用的派生方法:super().__init__() --super()的属性查找顺序是从当前位置开始找,根据mro列表,当前没有就往上找. super() ...

  3. Python中的super().__init__()

    Python里的super().init()有什么用? 简单的说super().__init__(),就是继承父类的init方法,同样可以使用super()去继承其他方法. 下面是三种不同的继承.调用 ...

  4. python中super().__init__和类名.__init__的区别

    super().__init__相对于类名.__init__,在单继承上用法基本无差 但在多继承上有区别,super方法能保证每个父类的方法只会执行一次,而使用类名的方法会导致方法被执行多次 多继承时 ...

  5. super().__init__()方法

    class first(object): def __init__(self,age,name): self.age = age self.name = name class second(first ...

  6. Python子类构造函数调用super().__init__()用法说明

    一.super的作用 1.如果子类(Puple)继承父类(Person)不做初始化,那么会自动继承父类(Person)属性name.2.如果子类(Puple_Init)继承父类(Person)做了初始 ...

  7. 论super().__init__()的用法

    当我们调用 super() 的时候,实际上是实例化了一个 super 类. super 是个类,既不是关键字也不是函数等其他数据结构,该对象就是专门用来访问父类中的属性的(严格按照继承的关系). -- ...

  8. 【python】类中的super().__init__()

    来源:百度知道 python中的super( test, self).__init__()是什么意思? 首先找到test的父类(比如是类A),然后把类test的对象self转换为类A的对象,然后&qu ...

  9. Python 面向对象编程的一些特征 及 单例模式的实现, 父类方法的调用(python2/python3)

    一.面向对象编程几个特征(封装, 继承,多态) 1.封装:类里面封装函数和变量, 在将类进行实例化成实例对象时进行传参, 从而生成不同的实例对象,增加代码的复用. 2.继承:子类可以继承父类的方法和属 ...

随机推荐

  1. 算法习题---4-9数据挖掘(Uva1591)

    一:题目 这是最懵逼的一道题,什么鬼......... [刷题]算法竞赛入门经典(第2版) 4-9/UVa1591 - Data Mining(详细题目看这个吧,不想多说) 二:代码实现 #defin ...

  2. 用python画 pareto front

    用python画 pareto front 觉得有用的话,欢迎一起讨论相互学习~Follow Me 2D pf import os import matplotlib.pyplot as plt im ...

  3. Linux系统调优——内核相关参数(五)

    修改内核参数有3种办法:一种临时修改,两种永久修改. 临时修改是使用sysctl [选项] [参数名=值]命令:永久修改是修改/etc/sysctl.conf文件或修改/proc/sys/目录下的对应 ...

  4. 关于PF_INET和AF_INET的区别

    在写网络程序的时候,建立TCP socket:   sock = socket(PF_INET, SOCK_STREAM, 0);然后在绑定本地地址或连接远程地址时需要初始化sockaddr_in结构 ...

  5. [转]Office 安装卸载太麻烦?用这个工具帮你解决:Office Tool Plus

    原文链接:https://sspai.com/post/43839 Office Tool官方网站:https://otp.landian.vip/zh-cn/ 真的很好用,发一个安装的截图:

  6. 【error】OutOfRangeError (see above for traceback): RandomShuffleQueue

    前言 在使用tensorflow TFRecord的过程中,读取*.tfrecord文件时出现错误,本文解决这个错误. 错误描述: OutOfRangeError (see above , curre ...

  7. .mmap文件如何打开

    .mmap是一种思维导图文件的后缀,可以用Xmind软件打开.

  8. 【译】Vue源码学习(一):Vue对象构造函数

    本系列文章详细深入Vue.js的源代码,以此来说明JavaScript的基本概念,尝试将这些概念分解到JavaScript初学者可以理解的水平.有关本系列的一些后续的计划和轨迹的更多信息,请参阅此文章 ...

  9. MySQL 过滤复制+复制映射 配置方法

    场景 node1 和 node2 为两台不同业务的MySQL服务器.业务方有个需求,需要将node1上的 employees库的departments .dept_manager 这2张表同步到 no ...

  10. Mac下安装VirtualBox并在VirtualBox中安装CentOS7

    VirtualBox (百科)VirtualBox 是一款开源虚拟机软件.VirtualBox 是由德国 Innotek 公司开发,由Sun Microsystems公司Sun Microsystem ...