魔术方法
简单说下魔术方法的概念,python 语言的一种语法特性,可以让一些函数不用被显式的调用的时候被执行。这种方法的设置是以__开头和结尾,如__init__,__furture__等。
构造函数
__init__ 是Python的构造函数,在python对象被创建的时候被执行,与之对应的是__del__析构函数,在对象生命周期结束的时候被执行,我们尝试构建一个构造函数:
class FooBar:
def __init__(self):
self.somevar = 42
>>> f =FooBar()
>>> f.somevar
4
在上述对象呗创建的时候,构造函数将会被自动执行。完成一些系统的初始化工作,构造函数是python使用频率最高的魔术方法。
构造函数同样可以被继承
class A:
def hello(self):
print 'hello . I am A.'
class B(A):
pass
>>> a = A()
>>> b = B()
>>> a.hello()
hello . I am A
自然也可以被重写
class A:
def hello(self):
print 'hello . I am A.'
class B(A):
def hello(self):
print 'hello . I am B'
>>> b = B()
>>> b.hello()
hello . I am B
但是在一些场景中,子类的构造函数重写的时候,需要对调用父类的构造函数
class Bird:
def __init__(self):
self.hungry = True
def eat(self):
if self.hungry:
print 'Aaaah...'
self.hungry = False
else:
print 'No, thanks!'
class SongBird(Bird):
def __init__(self):
self.sound = 'Squawk!'
def sing(self):
print self.sound
>>> s = SongBird()
>>> s.sing()
Squawk!
>>> s.eat()
Traceback (most recent call last):
File "<pyshell#26>", line 1, in <module>
s.eat()
File "C:/Python27/bird", line 6, in eat
if self.hungry:
AttributeError: 'SongBird' object has no attribute 'hungry'
异常很清楚地说明了错误:SongBird没有hungry特性。原因是这样的:在SongBird中,构造方法被重写,但新的构造方法没有任何关于初始化hungry特性的代码。为了达到预期的效果,SongBird的构造方法必须调用其超类Bird的构造方法来确保进行基本的初始化。
对于如何解决这个问题,我们使用super方法解决这个问题
__metaclass__ = type #表明为新式类
class Bird:
def __init__(self):
self.hungry = True
def eat(self):
if self.hungry:
print 'Aaaah...'
self.hungry = Falseit
else:
print 'No, thanks!'
class SongBird(Bird):
def __init__(self):
super(SongBird,self).__init__()
self.sound = 'Squawk!'
def sing(self):
print self.sound
>>> s.sing()
Squawk!
>>> s.eat()
Aaaah...
>>> s.eat()
No, thanks!
下面看下另外一类魔术方法
getitem setitem
我们使用一个叫test.py的脚本来解释一下,其中checkIndex函数是判断传入的key是否为合法key,检查是否有类型错误或者语法错误,有的话抛出异常。AS类包含getitem和__setitem
,他们分别在显示设置参数和获取参数的时候执行。下面第一段是代码内容
#!/bin/python
def checkIndex(key):
if not isinstance(key,(int,long)):raise TypeError
if key<0:raise indexError
class AS:
def __init__(self,start=0,step=1):
self.start = start
self.step = step
self.changed = {}
def __getitem__(self,key):
checkIndex(key)
print "getitem has been create"
try:return self.changed[key]
except KeyError:
return self.start+key*self.step
def __setitem__(self,key,value):
print "setitem has benn create"
checkIndex(key)
self.changed[key] = value
s = AS(1,2)
print s[4]
s[4]=2
print s[4]
print s[5]
执行结果如下:
getitem has been create
9
setitem has benn create
getitem has been create
2
getitem has been create
11
可以看到我们的程序并没有显示的调用这两个函数,但是他们在合适的时间自动执行了。