一、什么是描述符
本質(zhì)上看,描述符就是一個類,只不過它定義了另一個類中屬性的訪問方式。換句話說,一個類可以將屬性管理全權(quán)委托給描述符類。
描述符是 Python 中復(fù)雜屬性訪問的基礎(chǔ),它在內(nèi)部被用于實現(xiàn) property、方法、類方法、靜態(tài)方法和 super 類型。
描述符類基于以下 3 個特殊方法,換句話說,這 3 個方法組成了描述符協(xié)議:
__set__(self, obj, type=None):在設(shè)置屬性時將調(diào)用這一方法(本節(jié)后續(xù)用 setter 表示);__get__(self, obj, value):在讀取屬性時將調(diào)用這一方法(本節(jié)后續(xù)用 getter 表示);__delete__(self, obj):對屬性調(diào)用 del 時將調(diào)用這一方法。其中,實現(xiàn)了 setter 和 getter 方法的描述符類被稱為數(shù)據(jù)描述符;反之,如果只實現(xiàn)了 getter 方法,則稱為非數(shù)據(jù)描述符。
實際上,在每次查找屬性時,描述符協(xié)議中的方法都由類對象的特殊方法 __getattribute__() 調(diào)用(注意不要和 __getattr__() 弄混)。也就是說,每次使用類對象.屬性(或者 getattr(類對象,屬性值))的調(diào)用方式時,都會隱式地調(diào)用 __getattribute__(),它會按照下列順序查找該屬性:
驗證該屬性是否為類實例對象的數(shù)據(jù)描述符;如果不是,就查看該屬性是否能在類實例對象的 __dict__ 中找到;最后,查看該屬性是否為類實例對象的非數(shù)據(jù)描述符。為了表達清楚,這里舉個例子:
#描述符類class revealAccess:def __init__(self, initval = None, name = ‘var’):self.val = initvalself.name = namedef __get__(self, obj, objtype):print(“Retrieving”,self.name)return self.valdef __set__(self, obj, val):print(“updating”,self.name)self.val = valclass myClass:x = revealAccess(10,’var “x”‘)y = 5m = myClass()print(m.x)m.x = 20print(m.x)print(m.y)運行結(jié)果為: Retrieving var “x”
10
updating var “x”
Retrieving var “x”
20
5
從這個例子可以看到,如果一個類的某個屬性有數(shù)據(jù)描述符,那么每次查找這個屬性時,都會調(diào)用描述符的 __get__() 方法,并返回它的值;同樣,每次在對該屬性賦值時,也會調(diào)用 __set__() 方法。
注意,雖然上面例子中沒有使用 __del__() 方法,但也很容易理解,當(dāng)每次使用 del 類對象.屬性(或者 delattr(類對象,屬性))語句時,都會調(diào)用該方法。
延伸閱讀:
二、python運算符
算術(shù)運算符標(biāo)準(zhǔn)算術(shù)運算符:加+ 減- 乘* 除 / 整除//取余運算符:%冪運算符:**賦值運算符運算順序從右到左支持鏈?zhǔn)竭\算:一個內(nèi)存,多個引用支持參數(shù)賦值支持系列解包賦值:左側(cè)變量必須和右側(cè)值個數(shù)一致;可快速交換賦值(a,b=b,a)比較運算符比較運算符:對變量或者表達式的結(jié)果進行大小、真假等比較比較運算符的結(jié)果為bool類型布爾運算符and、or、not、in、not in位運算符將數(shù)據(jù)轉(zhuǎn)成二進制進行計算