对比Python中__getattr__和 __getattribute__获取属性的用法
|
相信大家觉得大多数时候我们并不太需要关注getattribute和getattr的一些细节(至少我自己吧:)), 不过作为一个有好奇心有追求有气质的python宝宝,怎么可能不稍稍研究一下呢。好吧,其实是在github上读到一个开源项目sinaweibopy的源码才看的,代码挺有意思,正好当作一个实用的例子,来看看如何自定义实现gettattr让代码更加的动态优雅:
# 例子在原来的基础上简化了一下,排除依赖和干扰,详细参见原项目
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的时候返回一个 示例
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,value):
return len
if __name__=="__main__":
test = Test('letian')
print test.getlength('letian')
运行结果: 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 深入思考 1. 用作实例属性的获取和拦截
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的时候防止无限递归
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方法,对于属性的查找是完全能用的
class AboutAttr(object):
def __init__(self,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. (编辑:安卓应用网) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |
