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

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

 

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

 

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

 

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.nameb
b.funcb()

b.funca()

 

结果:

 

bbb function b : bbb
Traceback (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.namea
AttributeError: B instance has no attribute 'namea'

 

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

 

 

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

修改代码,多增一行:

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.nameb
b.funcb()
b.funca()

 

方法二:使用super函数

 

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

 

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.nameb
b.funcb()

b.funca()

 

第一句让类A继承自object类,这样才能使用super函数,

第二句使用super函数

 

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

 

转自(http://www.crazyant.net/1303.html