str()返回用户看到的字符串,而repr()返回程序开发者看到的字符串,也就是说,repr()是为调试服务的。
但是通常str()和repr()代码都是一样的,所以,有个偷懒的写法:
class Student(object):
def __init__(self, name):
self.name = name
def __str__(self):
return 'Student object (name=%s)' % self.name
__repr__ = __str__
_iter_():
如果一个类像实现for—-in 循环,类似list,tuple那样,就必须实现一个iter()方法,该方法返回一个迭代对象
_getitem_():
要表现得像list那样按照下标取出元素,需要实现getitem()方法
原因是getitem()传入的参数可能是一个int,也可能是一个切片对象slice,所以要做判断:
class Fib(object):
def __getitem__(self, n):
if isinstance(n, int): n是索引
a, b = 1, 1
for x in range(n):
a, b = b, a + b
return a
if isinstance(n, slice): # n是切片
start = n.start
stop = n.stop
if start is None:
start = 0
a, b = 1, 1
L = []
for x in range(stop):
if x >= start:
L.append(a)
a, b = b, a + b
return L
此外,如果把对象看成dict,getitem()的参数也可能是一个可以作key的object,例如str。
与之对应的是setitem()方法,把对象视作list或dict来对集合赋值。最后,还有一个delitem()方法,用于删除某个元素。
总之,通过上面的方法,我们自己定义的类表现得和Python自带的list、tuple、dict没什么区别,这完全归功于动态语言的“鸭子类型”,不需要强制继承某个接口。
_getattr_()
python参考手册 第四版
在查找属性(如obj.name)时,将调用特殊方法obj.getattrribute(‘name’)【注:python官方文档为obj.getattrribute(self,’name’)】.该方法执行搜索来查找该属性,这通常涉及检查特性,查找局部dict属性,检查字典以及搜索基类。如果搜索过程失败,最终会尝试调用类的getattr()方法(如果已定义)来查找属性。如果失败,抛出AttributeError异常
用完全动态的getattr,我们可以写出一个链式调用:
class Chain(object):
def __init__(self, path=''):
self._path = path
def __getattr__(self, path):
return Chain('%s/%s' % (self._path, path))
def __str__(self):
return self._path
__repr__ = __str__
Chain().status.user.timeline.list
‘/status/user/timeline/list’
还有些REST API会把参数放到URL中,比如GitHub的API:
GET /users/:user/repos
调用时,需要把:user替换为实际用户名。如果我们能写出这样的链式调用:
Chain().users(‘michael’).repos
说明链式调用的内容
本人实现Chain().users(‘michael’).repos
class Chain(object):
def __init__(self, path=''):
self._path = path
def __getattr__(self, path):
return Chain('%s/%s' % (self._path, path))
def __str__(self):
return self._path
def user(self,name):
self._path = name
return self
__repr__ = __str__
Chain().user(‘google’).repos
google/repos
_call_():
任何类,只需要定义一个call()方法,就可以直接对实例进行调用。
过callable()函数,我们就可以判断一个对象是否是“可调用”对象。
廖雪峰python学习地址