Python/ target=_blank class=infotextkey>Python中,通過使用描述符,可以讓程序員在引用一個對象屬性時自定義要完成的工作。
本質(zhì)上看,描述符就是一個類,只不過它定義了另一個類中屬性的訪問方式。換句話說,一個類可以將屬性管理全權(quán)委托給描述符類。
描述符是Python中復雜屬性訪問的基礎(chǔ),它在內(nèi)部被用于實現(xiàn)property、方法、類方法、靜態(tài)方法和super類型。
描述符類基于以下3個特殊方法,換句話說,這3個方法組成了描述符協(xié)議:
__set__(self,obj,type=None):在設置屬性時將調(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__(),它會按照下列順序查找該屬性:
1、驗證該屬性是否為類實例對象的數(shù)據(jù)描述符;
2、如果不是,就查看該屬性是否能在類實例對象的__dict__中找到;
3、最后,查看該屬性是否為類實例對象的非數(shù)據(jù)描述符。
為了表達清楚,這里舉個例子:
#描述符類
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]
運行結(jié)果為:
Retrieving var "x"
10
updating var "x"
Retrieving var "x"
20
5123456復制代碼類型:[python]
從這個例子可以看到,如果一個類的某個屬性有數(shù)據(jù)描述符,那么每次查找這個屬性時,都會調(diào)用描述符的__get__()方法,并返回它的值;同樣,每次在對該屬性賦值時,也會調(diào)用__set__()方法。
注意,雖然上面例子中沒有使用__del__()方法,但也很容易理解,當每次使用del類對象.屬性(或者delattr(類對象,屬性))語句時,都會調(diào)用該方法。
除了使用描述符類自定義類屬性被調(diào)用時做的操作外,還可以使用property()函數(shù)或者@property裝飾器,它們會在后續(xù)章節(jié)做詳細介紹。