相信大家看到這個標題的時候也會立馬在腦海裡面過一遍,覺得大多數時候我們並不太需要關注getattribute
和getattr
的一些細節(至少我自己吧:)),
一般情況下消費我們自定義的類的時候,我們對類的結構都瞭解,不會刻意偏離,造成一些屬性訪問的錯誤。
不過作為一個有好奇心有追求有氣質的python寶寶,怎麼可能不稍稍研究一下呢。好吧,其實是在github上讀到一個開源專案sinaweibopy的原始碼才看的,程式碼挺有意思,正好當作一個實用的例子,來看看如何自定義實現gettattr
讓程式碼更加的動態優雅:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
# 例子在原來的基礎上簡化了一下,排除依賴和干擾,詳細參見原專案 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的介面啦)。
既然能通過定製類的getattr
自定義方法來實現一些優雅的功能,自然我們也要對它有一些瞭解,包括和它相似的自定義方法getattribute
1. 用作例項屬性的獲取和攔截
當訪問某個例項屬性時, getattribute會被無條件呼叫,如未實現自己的getattr方法,會丟擲AttributeError
提示找不到這個屬性,如果自定義了自己getattr方法的話,方法會在這種找不到屬性的情況下被呼叫,比如上面的例子中的情況。所以在找不到屬性的情況下通過實現自定義的getattr方法來實現一些功能是一個不錯的方式,因為它不會像getattribute方法每次都會呼叫可能會影響一些正常情況下的屬性訪問:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
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,造成迴圈呼叫:
1 2 3 4 5 6 7 8 9 |
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
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
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.
關於getattribute和getattr的特性與區別就扯到這,要更深入的瞭解可以自行google,編寫高質量程式碼:改善Python程式的91個建議這本書裡面也有一些詳細的介紹。
參考資料: