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

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()这一句执行

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值