python学习笔记3:面向对象的高级特性3 定制类
2017-03-10 20:44
495 查看
Python的class中有许多特殊用途的函数,可以帮助我们定制类。
请看代码:
运行结果:
__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()函数。
我们再来看一个复杂的例子,请仔细琢磨:
运行结果:
我是这样理解的,最后一句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()这一句执行
请看代码:
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()这一句执行
相关文章推荐
- python学习笔记1:面向对象的高级特性1__slots__
- 【Python学习笔记】python高级特性:迭代
- Python 学习笔记(三):文件,高级特性,枚举,正则表达式,模块
- Python学习笔记06_高级特性
- python学习笔记之函数及其高级特性
- 【Python学习笔记】python高级特性:列表生成式
- Python学习笔记(三)--高级特性
- 【Python】学习笔记——-4.0、高级特性
- python学习笔记-高级特性-切片(Slice)
- Python学习笔记--高级特性
- python2.7学习笔记(6) ——高级特性:切片、迭代、列表生成式、生成器
- Python高级特性——学习笔记
- Python学习笔记二:高级特性
- python学习笔记----高级特性
- 廖雪峰python学习笔记4:高级特性
- Python学习笔记——高级特性
- Python之学习笔记(高级特性)
- Python高级特性(切片 迭代 列表生成式 生成器 迭代器)学习笔记
- Python学习笔记——高级特性
- Python学习笔记 - 高级特性