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)
<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__()一样

原文地址:https://www.cnblogs.com/Peter2014/p/10844204.html

时间: 2024-10-10 17:44:03

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

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

""" 推荐使用的派生方法:super().__init__() --super(),严格继承属性查找顺序 --super(),会得到一个特殊的对象,该对象就是专门用来 访问父类中的属性的(严格按照继承的关系) --super().__init__(),此处__init__(), 括号中不需要传值self. --在Python2中,super()的完整用法是super(自己类名,self), 在Python2中需要写完整,而Python3中可以简写为super(). &quo

super().__init__()的用法

--super(),会得到一个特殊的对象,该对象就是专门用来 访问父类中的属性的(严格按照继承的关系) --super().__init__(),这里__init__()括号中不需要传参self, 有其他参数传其他参数即可 --在Python2中,super()的完整用法是 super(自己类名,self).__init__(), 在Python2中需要写完整,而Python3中可以简写为 super().__init__(). --用法解析:在子类的__init__方法中使用 super().

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

super().__init__相对于类名.__init__,在单继承上用法基本无差 但在多继承上有区别,super方法能保证每个父类的方法只会执行一次,而使用类名的方法会导致方法被执行多次 多继承时,使用super方法,对父类的传参数,应该是由于python中super的算法导致的原因,必须把参数全部传递,否则会报错 单继承时,使用super方法,则不能全部传递,只能传父类方法所需的参数,否则会报错 多继承时,相对于使用类名.__init__方法,要把每个父类全部写一遍, 而使用super方法

python-子类构造函数调用super().__init__()

[super作用] 如果子类(Puple)继承父类(Person)不做初始化,那么会自动继承父类(Person)属性name.如果子类(Puple_Init)继承父类(Person)做了初始化,且不调用super初始化父类构造函数,那么子类(Puple_Init)不会自动继承父类的属性(name).如果子类(Puple_super)继承父类(Person)做了初始化,且调用了super初始化了父类的构造函数,那么子类(Puple_Super)也会继承父类的(name)属性. 原文地址:https

python之路---面向对象编程(二)

类的继承 1.在python3中,只有新式类,新式类的继承方式为:广度优先.而python2中,经典类的继承方式为:深度优先.那么我们来看看深度优先和广度优先的区别吧 如下图,为类之间的继承关系.B,C继承A,D继承B,E继承C,F继承D,E   深度优先: class A: # def foo(self): # print 'from A' pass class B(A): # def foo(self): # print 'from B' pass class C(A): # def foo

Python3基础 super层层调用父类的__init__方法 子类的__init__覆盖了父类的__init__的解决方法

镇场诗: 诚听如来语,顿舍世间名与利.愿做地藏徒,广演是经阎浮提. 愿尽吾所学,成就一良心博客.愿诸后来人,重现智慧清净体.------------------------------------------ex1: code: class Parent : def __init__(self) : #父类的构造函数 print("父类构造完毕") class Child1(Parent) : #child类继承于 parent类 def __init__(self) : print(

Python3基础 super 子类调用父类的__init__

? python : 3.7.0 OS : Ubuntu 18.04.1 LTS IDE : PyCharm 2018.2.4 conda : 4.5.11 type setting : Markdown ? example_1 code """ @Author : 行初心 @Date : 18-9-23 @Blog : www.cnblogs.com/xingchuxin @GitHub : github.com/GratefulHeartCoder ""

python3 module中__init__.py的需要注意的地方

网上关于__init__.py的作用的资料到处都是,我在此就不再啰嗦哪些了. 若有需要.请各位看官去搜搜即可. 最近刚开始用Python3 就遇到了这个比较有意思的事情 闲言少叙,下面要介绍的是python3中__init__.py中的一些略微不同的地方 目录结构如下: ├── mdl │ ├── __init__.py │ ├── bar.py │ └── foo.py └── test.py 文件内容,从上至下,依次为 mdl/__init__.py: #encoding:utf-8 fro

python中的super

super用于类的继承.用super()代替父类名 (一)通过类名调用父类中的方法                                                         (二)通过super调用父类中的方法(python3) 结果如下: python3中的写法:super().__init__() python2中的写法:super(B,self).__init__()  B子类,不是父类,别写错了.