python中doc=parased.getroot()_对比Python中__getattr__和 __getattribute__获取属性的用法

# 例子在原来的基础上简化了一下,排除依赖和干扰,详细参见原项目

class UrlGenerator(object):

def __init__(self, root_url):

self.url = root_url

def __getattr__(self, item):

if item == 'get' or item == 'post':

print self.url

return UrlGenerator('{}/{}'.format(self.url, item))

url_gen = UrlGenerator('http://xxxx')

url_gen.users.show.get

>>> http://xxxx/users/show

充分利用getattr会在没有查找到相应实例属性时被调用的特点,方便的通过链式调用生成对应的url,源代码中在碰到http method的时候返回一个

可调用的对象更加的优雅,链式的操作不仅优雅而且还能很好的说明调用的接口的意义(restful的接口啦)。

示例1.__getattr__示例:

class Test(object):

def __init__(self,name):

self.name = name

def __getattr__(self, value):

if value == 'address':

return 'China'

if __name__=="__main__":

test = Test('letian')

print test.name

print test.address

test.address = 'Anhui'

print test.address

运行结果:

letian

China

Anhui

如果是调用了一个类中未定义的方法,则__getattr__也要返回一个方法,例如:

class Test(object):

def __init__(self,name):

self.name = name

def __getattr__(self, value):

return len

if __name__=="__main__":

test = Test('letian')

print test.getlength('letian')

运行结果:

6

2.__getattribute__示例:

class Test(object):

def __init__(self,name):

self.name = name

def __getattribute__(self, value):

if value == 'address':

return 'China'

if __name__=="__main__":

test = Test('letian')

print test.name

print test.address

test.address = 'Anhui'

print test.address

运行结果:

None

China

China

深入思考既然能通过定制类的getattr自定义方法来实现一些优雅的功能,自然我们也要对它有一些了解,包括和它相似的自定义方法getattribute

1. 用作实例属性的获取和拦截当访问某个实例属性时, getattribute会被无条件调用,如未实现自己的getattr方法,会抛出AttributeError提示找不到这个属性,如果自定义了自己getattr方法的话,方法会在这种找不到属性的情况下被调用,比如上面的例子中的情况。所以在找不到属性的情况下通过实现自定义的getattr方法来实现一些功能是一个不错的方式,因为它不会像getattribute方法每次都会调用可能会影响一些正常情况下的属性访问:

class Test(object):

def __init__(self, p):

self.p = p

def __getattr__(self, item):

return 'default'

t = Test('p1')

print t.p

print t.p2

>>> p1

>>> default

2. 自定义getattribute的时候防止无限递归因为getattribute在访问属性的时候一直会被调用,自定义的getattribute方法里面同时需要返回相应的属性,通过self.__dict__取值会继续向下调用getattribute,造成循环调用:

class AboutAttr(object):

def __init__(self, name):

self.name = name

def __getattribute__(self, item):

try:

return super(AboutAttr, self).__getattribute__(item)

except KeyError:

return 'default'

这里通过调用绑定的super对象来获取队形的属性,对新式类来说其实和object.__getattribute__(self, item)一样的道理:

默认情况下自定义的类会从object继承getattribute方法,对于属性的查找是完全能用的

getattribute的实现感觉还是挺抽象化的,只需要绑定相应的实例对象和要查找的属性名称就行

3.同时覆盖掉getattribute和getattr的时候,在getattribute中需要模仿原本的行为抛出AttributeError或者手动调用getattr

class AboutAttr(object):

def __init__(self, name):

self.name = name

def __getattribute__(self, item):

try:

return super(AboutAttr, self).__getattribute__(item)

except KeyError:

return 'default'

except AttributeError as ex:

print ex

def __getattr__(self, item):

return 'default'

at = AboutAttr('test')

print at.name

print at.not_exised

>>>test

>>>'AboutAttr' object has no attribute 'not_exised'

>>>None

上面例子里面的getattr方法根本不会被调用,因为原本的AttributeError被我们自行处理并未抛出,也没有手动调用getattr,所以访问not_existed的结果是None而不是default.

f68f2add0b68e4f9810432fce46917b7.png

本文原创发布php中文网,转载请注明出处,感谢您的尊重!

相关文章

相关视频

网友评论

文明上网理性发言,请遵守 新闻评论服务协议我要评论

47d507a036d4dd65488c445c0974b649.png

立即提交

专题推荐064df72cb40df78e80e61b7041ee044f.png独孤九贱-php全栈开发教程

全栈 100W+

主讲:Peter-Zhu 轻松幽默、简短易学,非常适合PHP学习入门

7dafe36c040e31d783922649aefe0be1.png玉女心经-web前端开发教程

入门 50W+

主讲:灭绝师太 由浅入深、明快简洁,非常适合前端学习入门

04246fdfe8958426b043c89ded0857f1.png天龙八部-实战开发教程

实战 80W+

主讲:西门大官人 思路清晰、严谨规范,适合有一定web编程基础学习

php中文网:公益在线php培训,帮助PHP学习者快速成长!

Copyright 2014-2020 https://www.php.cn/ All Rights Reserved | 苏ICP备2020058653号-1e6cebb680dfe320dad7e62bd6442c3a6.gif

4b6782251ac158858f67da1fce837421.png  

2cf0d7eb6d8ded0d697a2f14e9addeba.png

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值