专栏首页python3python中子类调用父类的初始化方法

python中子类调用父类的初始化方法

http://bestchenwu.iteye.com/blog/1044848

http://www.crazyant.net/1303.html

一直不太理解python的初始化方法,今天找了下资料,先放这改天整理下

一、第一种解释:

python和其他面向对象语言类似,每个类可以拥有一个或者多个父类,它们从父类那里继承了属性和方法。如果一个方法在子类的实例中被调用,或者一个属性在子类的实例中被访问,但是该方法或属性在子类中并不存在,那么就会自动的去其父类中进行查找。

继承父类后,就能调用父类方法和访问父类属性,而要完成整个集成过程,子类是需要调用的构造函数的。

子类不显式调用父类的构造方法,而父类构造函数初始化了一些属性,就会出现问题

如果子类和父类都有构造函数,子类其实是重写了父类的构造函数,如果不显式调用父类构造函数,父类的构造函数就不会被执行,导致子类实例访问父类初始化方法中初始的变量就会出现问题。

12345678910111213141516171819

class A:    def __init__(self):        self.namea="aaa"     def funca(self):        print "function a : %s"%self.namea class B(A):    def __init__(self):        self.nameb="bbb"     def funcb(self):        print "function b : %s"%self.nameb b=B()print b.namebb.funcb() b.funca()

结果:

12345678

bbbfunction b : bbbTraceback (most recent call last):  File "D:\workbench\python\MyPythonProject\test\study\overwrite_method.py", line 19, in <module>    print b.funca()  File "D:\workbench\python\MyPythonProject\test\study\overwrite_method.py", line 6, in funca    print "function a : %s"%self.nameaAttributeError: B instance has no attribute 'namea'

在子类中,构造函数被重写,但新的构造方法没有任何关于初始化父类的namea属性的代码,为了达到预期的效果,子类的构造方法必须调用其父类的构造方法来进行基本的初始化。有两种方法能达到这个目的:调用超类构造方法的未绑定版本,或者使用super函数。

方法一:调用未绑定的超类构造方法

修改代码,多增一行:

12345678910111213141516171819202122

class A:    def __init__(self):        self.namea="aaa"     def funca(self):        print "function a : %s"%self.namea class B(A):    def __init__(self):        #这一行解决了问题        A.__init__(self)        self.nameb="bbb"     def funcb(self):        print "function b : %s"%self.nameb b=B()print b.namebb.funcb() b.funca()

如上有注释的一行解决了该问题,直接使用父类名称调用其构造函数即可。

这种方法叫做调用父类的未绑定的构造方法。在调用一个实例的方法时,该方法的self参数会被自动绑定到实例上(称为绑定方法)。但如果直接调用类 的方法(比如A.__init),那么就没有实例会被绑定。这样就可以自由的提供需要的self参数,这种方法称为未绑定unbound方法。

通过将当前的实例作为self参数提供给未绑定方法,B类就能使用其父类构造方法的所有实现,从而namea变量被设置。

方法二:使用super函数

修改代码,这次需要增加在原来代码上增加2行:

12345678910111213141516171819202122

#父类需要继承object对象class A(object):    def __init__(self):        self.namea="aaa"     def funca(self):        print "function a : %s"%self.namea class B(A):    def __init__(self):        #这一行解决问题        super(B,self).__init__()        self.nameb="bbb"     def funcb(self):        print "function b : %s"%self.nameb b=B()print b.namebb.funcb() b.funca()

如上有注释的为新增的代码,其中第一句让类A继承自object类,这样才能使用super函数,因为这是python的“新式类”支持的特性。当前的雷和对象可以作为super函数的参数使用,调用函数返回的对象的任何方法都是调用超类的方法,而不是当前类的方法。

super函数会返回一个super对象,这个对象负责进行方法解析,解析过程其会自动查找所有的父类以及父类的父类。

 方法一更直观,方法二可以一次初始化所有超类

super函数比在超累中直接调用未绑定方法更直观,但是其最大的有点是如果子类继承了多个父类,它只需要使用一次super函数就可以。然而如果没有这个需求,直接使用A.__init__(self)更直观一些。

二、第二种解释

    在Python中子类继承父类的过程中,如果子类不覆盖父类的__init__()方法,则子类默认将执行与父类一样的初始化方法。但是假如子类自己重写 了(也成为覆盖)父类的__init__()方法,那么就需要显式的调用父类的初始化方法了。有两种方法可以做到:

     1:ParentClass.__init__(),父类名加上init函数

     2:super(type,cls).__init__() 

      重点介绍下这个,这个也是Python在借鉴了C++和JAVA的经验后,做出的改进语言熟悉程度的一种努力。

     super(type,cls)实质上是super类的3个静态方法之一。

     参考下super类的定义:

Python代码  

  1. class super(object)   
  2. super(type) -> unbound super object  
  3. super(type, obj) -> bound super object; requires isinstance(obj, type)  
  4. super(type, type2) -> bound super object; requires issubclass(type2, type)  
  5. Typical use to call a cooperative superclass method:  
  6. class C(B):  
  7. def meth(self, arg):  
  8.         super(C, self).meth(arg)  

    从图上可以看出,我们实质上在调用的是第二个方法,其中第一个参数是开始寻找父类的起始点(起始但不包括),第二个参数是需要一个对应第一个type的实 例,即满足isinstance(obj,type),这个方法将返回第一个满足继承关系的类,寻找顺序遵从type.__mro__属性顺序(不妨将它 认为是倒树的广度优先遍历顺序)。

   这个方法的好处在于将类之间的构造方法调用的耦合关系彻底解耦了。如果改变了当前Class,只需要适当的改变下super的第一个参数即可。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Python学习 :面向对象 -- 三大

          -- 子类会根据左侧优先原则,在第一个父类中寻找方法,在找不到的情况下在去父类的父类中寻找

    py3study
  • Python面向对象——继承

    py3study
  • Python_类的继承

    1.类的继承关系和生活中父亲、儿子、孙子之间的关系一样,Python中若A类继承B类,则A类称之为子类,B类称之为父类(也称为基类)。 2.类的继承方式分为:单...

    py3study
  • Python学习 :面向对象 -- 三大

          -- 子类会根据左侧优先原则,在第一个父类中寻找方法,在找不到的情况下在去父类的父类中寻找

    py3study
  • Python_类的继承

    1.类的继承关系和生活中父亲、儿子、孙子之间的关系一样,Python中若A类继承B类,则A类称之为子类,B类称之为父类(也称为基类)。 2.类的继承方式分为:单...

    py3study
  • Python_子类调用父类的方法

    1.方式一 子类调用父类的方法,包含2中形式的调用。一种形式是在类内部通过继承的方式调用父类的方法,另外一种形式是子类实例化后之后通过继承的方式来调用父类的方法...

    py3study
  • GitLab部署及常用命令

    GitLab社区版是免费的,不但能建立免费的私有仓库而且没有数量上限,参与人员也没有数量限制,还能设置成员的权限,甚至细致到具体某条分支的权限,以及强大的工作流...

    苦叶子
  • 关于mybatis的思考(1)——mybatis的使用实例

    Mister24
  • 爬虫养成记--千军万马来相见(详解多线程)

    本文由图雀社区[1]成员 灿若星空[2] 写作而成,欢迎加入图雀社区,一起创作精彩的免费技术教程,予力编程行业发展。

    一只图雀
  • 记忆(缓存)函数返回值:Python

    对于经常调用的函数,特别是递归函数或计算密集的函数,记忆(缓存)返回值可以显着提高性能。而在 Python 里,可以使用字典来完成。

    py3study

扫码关注云+社区

领取腾讯云代金券