python魔术方法

魔术方法

简单说下魔术方法的概念,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 "", line 1, in 
    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

可以看到我们的程序并没有显示的调用这两个函数,但是他们在合适的时间自动执行了。

你可能感兴趣的:(python魔术方法)