您的位置:首页 > 编程语言 > Python开发

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

2017-03-10 20:44 495 查看
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()这一句执行
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息