Python/ target=_blank class=infotextkey>Python中,通過使用描述符,可以讓程序員在引用一個對象屬性時自定義要完成的工作。
本質上看,描述符就是一個類,只不過它定義了另一個類中屬性的訪問方式。換句話說,一個類可以將屬性管理全權委托給描述符類。
描述符是Python中復雜屬性訪問的基礎,它在內部被用于實現property、方法、類方法、靜態方法和super類型。
描述符類基于以下3個特殊方法,換句話說,這3個方法組成了描述符協議:
__set__(self,obj,type=None):在設置屬性時將調用這一方法(本節后續用setter表示);
__get__(self,obj,value):在讀取屬性時將調用這一方法(本節后續用getter表示);
__delete__(self,obj):對屬性調用del時將調用這一方法。
其中,實現了setter和getter方法的描述符類被稱為數據描述符;反之,如果只實現了getter方法,則稱為非數據描述符。
實際上,在每次查找屬性時,描述符協議中的方法都由類對象的特殊方法__getattribute__()調用(注意不要和__getattr__()弄混)。也就是說,每次使用類對象.屬性(或者getattr(類對象,屬性值))的調用方式時,都會隱式地調用__getattribute__(),它會按照下列順序查找該屬性:
1、驗證該屬性是否為類實例對象的數據描述符;
2、如果不是,就查看該屬性是否能在類實例對象的__dict__中找到;
3、最后,查看該屬性是否為類實例對象的非數據描述符。
為了表達清楚,這里舉個例子:
#描述符類
class revealAccess:
def __init__(self, initval = None, name = 'var'):
self.val = initval
self.name = name
def __get__(self, obj, objtype):
print("Retrieving",self.name)
return self.val
def __set__(self, obj, val):
print("updating",self.name)
self.val = val
class myClass:
x = revealAccess(10,'var "x"')
y = 5
m = myClass()
print(m.x)
m.x = 20
print(m.x)
print(m.y)12345678910111213141516171819復制代碼類型:[python]
運行結果為:
Retrieving var "x"
10
updating var "x"
Retrieving var "x"
20
5123456復制代碼類型:[python]
從這個例子可以看到,如果一個類的某個屬性有數據描述符,那么每次查找這個屬性時,都會調用描述符的__get__()方法,并返回它的值;同樣,每次在對該屬性賦值時,也會調用__set__()方法。
注意,雖然上面例子中沒有使用__del__()方法,但也很容易理解,當每次使用del類對象.屬性(或者delattr(類對象,屬性))語句時,都會調用該方法。
除了使用描述符類自定義類屬性被調用時做的操作外,還可以使用property()函數或者@property裝飾器,它們會在后續章節做詳細介紹。