国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

對比Python中__getattr__和 __getattribute__獲取屬性的用法

2019-11-25 16:42:29
字體:
供稿:網(wǎng)友

相信大家覺得大多數(shù)時候我們并不太需要關(guān)注getattribute和getattr的一些細(xì)節(jié)(至少我自己吧:)),
一般情況下消費我們自定義的類的時候,我們對類的結(jié)構(gòu)都了解,不會刻意偏離,造成一些屬性訪問的錯誤。

不過作為一個有好奇心有追求有氣質(zhì)的python寶寶,怎么可能不稍稍研究一下呢。好吧,其實是在github上讀到一個開源項目sinaweibopy的源碼才看的,代碼挺有意思,正好當(dāng)作一個實用的例子,來看看如何自定義實現(xiàn)gettattr讓代碼更加的動態(tài)優(yōu)雅:

# 例子在原來的基礎(chǔ)上簡化了一下,排除依賴和干擾,詳細(xì)參見原項目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會在沒有查找到相應(yīng)實例屬性時被調(diào)用的特點,方便的通過鏈?zhǔn)秸{(diào)用生成對應(yīng)的url,源代碼中在碰到http method的時候返回一個
可調(diào)用的對象更加的優(yōu)雅,鏈?zhǔn)降牟僮鞑粌H優(yōu)雅而且還能很好的說明調(diào)用的接口的意義(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

運行結(jié)果:

letianChinaAnhui

如果是調(diào)用了一個類中未定義的方法,則__getattr__也要返回一個方法,例如:

class Test(object):  def __init__(self,name):    self.name = name  def __getattr__(self, value):    return lenif __name__=="__main__":  test = Test('letian')  print test.getlength('letian')

運行結(jié)果:
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

運行結(jié)果:

NoneChinaChina

深入思考
既然能通過定制類的getattr自定義方法來實現(xiàn)一些優(yōu)雅的功能,自然我們也要對它有一些了解,包括和它相似的自定義方法getattribute

1. 用作實例屬性的獲取和攔截
當(dāng)訪問某個實例屬性時, getattribute會被無條件調(diào)用,如未實現(xiàn)自己的getattr方法,會拋出AttributeError提示找不到這個屬性,如果自定義了自己getattr方法的話,方法會在這種找不到屬性的情況下被調(diào)用,比如上面的例子中的情況。所以在找不到屬性的情況下通過實現(xiàn)自定義的getattr方法來實現(xiàn)一些功能是一個不錯的方式,因為它不會像getattribute方法每次都會調(diào)用可能會影響一些正常情況下的屬性訪問:

class Test(object):  def __init__(self, p):    self.p = p  def __getattr__(self, item):    return 'default't = Test('p1')print t.pprint t.p2>>> p1>>> default

2. 自定義getattribute的時候防止無限遞歸
因為getattribute在訪問屬性的時候一直會被調(diào)用,自定義的getattribute方法里面同時需要返回相應(yīng)的屬性,通過self.__dict__取值會繼續(xù)向下調(diào)用getattribute,造成循環(huán)調(diào)用:

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'

這里通過調(diào)用綁定的super對象來獲取隊形的屬性,對新式類來說其實和object.__getattribute__(self, item)一樣的道理:

默認(rèn)情況下自定義的類會從object繼承g(shù)etattribute方法,對于屬性的查找是完全能用的
getattribute的實現(xiàn)感覺還是挺抽象化的,只需要綁定相應(yīng)的實例對象和要查找的屬性名稱就行
3.同時覆蓋掉getattribute和getattr的時候,在getattribute中需要模仿原本的行為拋出AttributeError或者手動調(diào)用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.nameprint at.not_exised>>>test>>>'AboutAttr' object has no attribute 'not_exised'>>>None

上面例子里面的getattr方法根本不會被調(diào)用,因為原本的AttributeError被我們自行處理并未拋出,也沒有手動調(diào)用getattr,所以訪問not_existed的結(jié)果是None而不是default.

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 北流市| 唐河县| 富川| 南平市| 方山县| 桂东县| 沅陵县| 嵊泗县| 亚东县| 通化市| 新乡县| 镇坪县| 岗巴县| 临沧市| 永德县| 磐安县| 普兰店市| 北安市| 磴口县| 米泉市| 镇平县| 遵义县| 高邮市| 拉萨市| 洛南县| 自贡市| 大化| 大荔县| 屏东市| 台安县| 营口市| 黑水县| 江达县| 湖南省| 绩溪县| 西宁市| 长海县| 阿拉善右旗| 翼城县| 如东县| 全椒县|