輕松理解Python 中的 descriptor
定義
通常,一個 descriptor 是具有“綁定行為”的對象屬性。所綁定行為可通過 descriptor 協(xié)議被自定義的 __get__() , __set__() 和 __delete__() 方法重寫。如果一個對象的上述三個方法任意一個被重寫,則就可被稱為 descriptor。
屬性的默認操作是從對象字典中獲取、設置和刪除一個屬性。例如,a.x 有一個查找鏈,先 a.__dict__['x'] ,若沒有則 type(a).__dict__['x'] ,若沒有增往上查找父類直到元類。如果查找鏈中,對象被定義了 descriptor 方法,Python 就會覆蓋默認行為。
Descriptor 是一個強大的工具,雖然開發(fā)者不常接觸到它,但它其實就是類、屬性、函數(shù)、方法、靜態(tài)方法、類方法以及 super() 背后的運行機制。
Descriptor 協(xié)議
三個方法原型如下所示:
descr.__get__(self, obj, type=None) --> value descr.__set__(self, obj, value) --> None descr.__delete__(self, obj) --> None
數(shù)據(jù) descriptor 是同時具有 __get__() 和 __set__() 方法的對象,若只有 __get__() 方法,則為非數(shù)據(jù) descriptor。如果實例字典中有和數(shù)據(jù) descriptor 同名的入口,則數(shù)據(jù) descriptor 優(yōu)先級更高。相反,非數(shù)據(jù) descriptor 優(yōu)先級低。
讓 __set__() 方法拋出異常,就能創(chuàng)建一個只讀數(shù)據(jù) descriptor。
調用 descriptor
descriptor 可以直接通過方法名調用。例如, d.__get__(obj) 。
而通過訪問對象屬性,自動調用 descriptor 才是更通用的做法。例如,如果 d 定義了方法 __get__() ,則 obj.d 會調用 d.__get__(obj) 。
對于對象, b.x 會被轉換成 type(b).__dict__['x'].__get__(b, type(b)) 。而對于類(是的,類也可以調用), B.x 會被轉換成 B.__dict__['x'].__get__(None, B) 。
Descriptor 例子
class RevealAccess(object):
"""A data descriptor that sets and returns values
normally and prints a message logging their access.
"""
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(object):
... x = RevealAccess(10, 'var "x"')
... y = 5
...
>>> m = MyClass()
>>> m.x
Retrieving var "x"
10
>>> m.x = 20
Updating var "x"
>>> m.x
Retrieving var "x"
20
>>> m.y
5
總結
以上所述是小編給大家介紹的Python 中的 descriptor,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網站的支持!
相關文章
一文深入學習Python中的os.listdir函數(shù)
這篇文章主要給大家介紹了關于Python中os.listdir函數(shù)的相關資料,os.listdir是 Python中的一個函數(shù),它的意思是返回指定目錄下的文件和文件夾的名稱的列表,需要的朋友可以參考下2023-10-10
Python真題案例之小學算術?階乘精確值?孿生素數(shù)?6174問題詳解
這篇文章主要介紹了python實操案例練習,本文給大家分享的案例中主要任務有小學生算術、階乘的精確值、孿生素數(shù)、6174問題,需要的小伙伴可以參考一下2022-03-03
NumPy實現(xiàn)從已有的數(shù)組創(chuàng)建數(shù)組
本文介紹了NumPy中如何從已有的數(shù)組創(chuàng)建數(shù)組,包括使用numpy.asarray,numpy.frombuffer和numpy.fromiter方法,具有一定的參考價值,感興趣的可以了解一下2024-10-10
django下創(chuàng)建多個app并設置urls方法
在本篇文章里小編給大家分享的是一篇關于django下創(chuàng)建多個app并設置urls方法,需要的朋友們可以參考學習下。2020-08-08

