Python的class中有许多特殊用途的函数,可以帮助我们定制类。
请看代码:
class Student(object):
def __init__(self,name):
self.name = name
print('创建了一个 %s' % self.name)
def __str__(self):
return 'Student object name:%s' % self.name
__repr__ = __str__
def __getattr__(self, attr):
return 'ByeBye'
def __call__(self):
print('My name is %s.' % self.name)
s = Student('HSX')
print(s)
print(Student('abc'))
print(s.age)
s()
Student('hsc')()
运行结果:
创建了一个 HSX
Student object name:HSX
创建了一个 abc
Student object name:abc
ByeBye
My name is HSX.
创建了一个 hsc
My name is hsc.
[Finished in 0.1s]
__str (双下划线):
定义这个函数之后输入类名可以调用此函数,print(s) 、print(Student(‘abc’))这两句都会调用str(self)函数
__getattr (双下划线):
定义这个函数后,当调用类没有的属性或方法时,就会调用这个函数,print(s.age)这一句就调用了这个函数。需要注意的是,getattr()函数的返回值可以是一个函数,比如用lambda表达式作为返回值。
__call(双下划线):
这个函数与str()函数类似,区别在于,这个函数是把对象当做函数来调用,s()、
Student(‘hsc’)()这两句调用call()函数,请注意与调用str()函数时的写法
还有需要注意的是s = Student(‘HSX’)、print(Student(‘abc’))、Student(‘hsc’)()都调用了init()函数。
我们再来看一个复杂的例子,请仔细琢磨:
class Chain(object):
def __init__(self, path='GET '):
self._path = path
print('init(%s) self._path=%s'%(path,self._path))
def __getattr__(self, path):
print('getattr(%s) self._path=%s'%(path,self._path))
return Chain('%s/%s' % (self._path,path))
def __call__(self,path):
print('call(%s) self._path=%s'%(path,self._path))
return Chain('%s/%s' % (self._path, path))
def __str__(self):
print('str() self._path=%s'%self._path)
return self._path
__repr__ = __str__
print(Chain().users('michael').repos)
运行结果:
init(GET ) self._path=GET
getattr(users) self._path=GET
init(GET /users) self._path=GET /users
call(michael) self._path=GET /users
init(GET /users/michael) self._path=GET /users/michael
getattr(repos) self._path=GET /users/michael
init(GET /users/michael/repos) self._path=GET /users/michael/repos
str() self._path=GET /users/michael/repos
GET /users/michael/repos
[Finished in 0.2s]
我是这样理解的,最后一句Chain().users(‘michael’).repos
首先Chain()会调用init()函数,path等于默认值;
Chain().users会调用getattr()函数,返回一个值,将返回值与原句剩下的拼起来,原句变成了:
Chain(‘%s/%s’ % (self._path,path))(‘michael’).repos,
这一句中的Chain(‘%s/%s’ % (self._path,path))又会调用init()函数,之后
Chain(‘%s/%s’ % (self._path,path))(‘michael’)类似于上面的例子中的Student(‘hsc’)(),会调用call()函数,返回一个值,原句变成
Chain(‘%s/%s’ % (self._path, path)).repos,先调用init()函数,在调用getattr()函数,原句变成:
Chain(‘%s/%s’ % (self._path,path)) 调用init函数,在调用str()函数
最后,print()这一句执行