分类: Python/Ruby
2014-07-09 14:20:25
如果子类和父类都有构造函数,子类其实是重写了父类的构造函数,如果不显式调用父类构造函数,父类的构造函数就不会被执行,
导致子类实例访问父类初始化方法中初始的变量就会出现问题。
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
Result:
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
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函数。
如上有注释的一行解决了该问题,直接使用父类名称调用其构造函数即可。
这种方法叫做调用父类的未绑定的构造方法。在调用一个实例的方法时,
该方法的self参数会被自动绑定到实例上(称为绑定方法)。
但如果直接调用类的方法(比如A.__init),那么就没有实例会被绑定。
这样就可以自由的提供需要的self参数,这种方法称为未绑定unbound方法。
通过将当前的实例作为self参数提供给未绑定方法,B类就能使用其父类构造方法的所有实现,
从而namea变量被设置。
方法二:使用super函数
修改代码,这次需要增加在原来代码上增加2行
#父类需要继承object对象