python学习笔记3:面向对象的高级特性3 定制类

来源:互联网 发布:妙味jquery源码分析 编辑:程序博客网 时间:2024/06/05 18:38

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')()

运行结果:

创建了一个 HSXStudent object name:HSX创建了一个 abcStudent object name:abcByeByeMy name is HSX.创建了一个 hscMy 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 /userscall(michael) self._path=GET /usersinit(GET /users/michael) self._path=GET /users/michaelgetattr(repos) self._path=GET /users/michaelinit(GET /users/michael/repos) self._path=GET /users/michael/reposstr() self._path=GET /users/michael/reposGET /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()这一句执行

1 1
原创粉丝点击