国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

深入解析Python中的descriptor描述器的作用及用法

2019-11-25 16:41:37
字體:
供稿:網(wǎng)友

一般來說,一個(gè)描述器是一個(gè)有“綁定行為”的對(duì)象屬性(object attribute),它的訪問控制被描述器協(xié)議方法重寫。這些方法是 __get__(), __set__(), 和 __delete__() 。有這些方法的對(duì)象叫做描述器。

默認(rèn)對(duì)屬性的訪問控制是從對(duì)象的字典里面(__dict__)中獲取(get), 設(shè)置(set)和刪除(delete)它。舉例來說, a.x 的查找順序是, a.__dict__['x'] , 然后 type(a).__dict__['x'] , 然后找 type(a) 的父類(不包括元類(metaclass)).如果查找到的值是一個(gè)描述器, Python就會(huì)調(diào)用描述器的方法來重寫默認(rèn)的控制行為。這個(gè)重寫發(fā)生在這個(gè)查找環(huán)節(jié)的哪里取決于定義了哪個(gè)描述器方法。注意, 只有在新式類中時(shí)描述器才會(huì)起作用。(新式類是繼承自 type 或者 object 的類)

描述器是強(qiáng)大的,應(yīng)用廣泛的。描述器正是屬性, 實(shí)例方法, 靜態(tài)方法, 類方法和 super 的背后的實(shí)現(xiàn)機(jī)制。描述器在Python自身中廣泛使用,以實(shí)現(xiàn)Python 2.2中引入的新式類。描述器簡(jiǎn)化了底層的C代碼,并為Python的日常編程提供了一套靈活的新工具。

描述器協(xié)議

descr.__get__(self, obj, type=None) --> valuedescr.__get__(self, obj, value) --> Nonedescr.__delete__(self, obj) --> None

一個(gè)對(duì)象如果是一個(gè)描述器,被當(dāng)做對(duì)象屬性(很重要)時(shí)重寫默認(rèn)的查找行為。

如果一個(gè)對(duì)象同時(shí)定義了__get__和__set__,它叫data descriptor。僅定義了__get__的描述器叫non-data descriptor。

data descriptor和non-data descriptor區(qū)別在于: 相對(duì)于實(shí)例的字典的優(yōu)先級(jí),如果實(shí)例字典有與描述器具同名的屬性,如果描述器是data descriptor,優(yōu)先使用data descriptor。如果是non-data descriptor,優(yōu)先使用字典中的屬性。

class B(object):  def __init__(self):    self.name = 'mink'  def __get__(self, obj, objtype=None):    return self.nameclass A(object):  name = B()a = A()print a.__dict__  # print {}print a.name    # print minka.name = 'kk'    print a.__dict__  # print {'name': 'kk'}print a.name    # print kk

這里B是一個(gè)non-data descriptor所以當(dāng)a.name = 'kk'的時(shí)候,a.__dict__里會(huì)有name屬性, 接下來給它設(shè)置__set__

def __set__(self, obj, value):  self.name = value ... do somethinga = A()print a.__dict__  # print {}print a.name    # print minka.name = 'kk'    print a.__dict__  # print {}print a.name    # print kk

因?yàn)閐ata descriptor訪問屬性優(yōu)先級(jí)比實(shí)例的字典高,所以a.__dict__是空的。

描述器的調(diào)用
描述器可以直接這么調(diào)用: d.__get__(obj)

然而更常見的情況是描述器在屬性訪問時(shí)被自動(dòng)調(diào)用。舉例來說, obj.d 會(huì)在 obj 的字典中找 d ,如果 d 定義了 __get__ 方法,那么 d.__get__(obj) 會(huì)依據(jù)下面的優(yōu)先規(guī)則被調(diào)用。

調(diào)用的細(xì)節(jié)取決于 obj 是一個(gè)類還是一個(gè)實(shí)例。另外,描述器只對(duì)于新式對(duì)象和新式類才起作用。繼承于 object 的類叫做新式類。

對(duì)于對(duì)象來講,方法 object.__getattribute__() 把 b.x 變成 type(b).__dict__['x'].__get__(b, type(b)) 。具體實(shí)現(xiàn)是依據(jù)這樣的優(yōu)先順序:資料描述器優(yōu)先于實(shí)例變量,實(shí)例變量?jī)?yōu)先于非資料描述器,__getattr__()方法(如果對(duì)象中包含的話)具有最低的優(yōu)先級(jí)。完整的C語言實(shí)現(xiàn)可以在 Objects/object.c 中 PyObject_GenericGetAttr() 查看。

對(duì)于類來講,方法 type.__getattribute__() 把 B.x 變成 B.__dict__['x'].__get__(None, B) 。用Python來描述就是:

def __getattribute__(self, key):  "Emulate type_getattro() in Objects/typeobject.c"  v = object.__getattribute__(self, key)  if hasattr(v, '__get__'):    return v.__get__(None, self)  return v

其中重要的幾點(diǎn):

  • 描述器的調(diào)用是因?yàn)?__getattribute__()
  • 重寫 __getattribute__() 方法會(huì)阻止正常的描述器調(diào)用
  • __getattribute__() 只對(duì)新式類的實(shí)例可用
  • object.__getattribute__() 和 type.__getattribute__() 對(duì) __get__() 的調(diào)用不一樣
  • 資料描述器總是比實(shí)例字典優(yōu)先。
  • 非資料描述器可能被實(shí)例字典重寫。(非資料描述器不如實(shí)例字典優(yōu)先)
  • super() 返回的對(duì)象同樣有一個(gè)定制的 __getattribute__() 方法用來調(diào)用描述器。調(diào)用 super(B, obj).m() 時(shí)會(huì)先在 obj.__class__.__mro__ 中查找與B緊鄰的基類A,然后返回 A.__dict__['m'].__get__(obj, A) 。如果不是描述器,原樣返回 m 。如果實(shí)例字典中找不到 m ,會(huì)回溯繼續(xù)調(diào)用 object.__getattribute__() 查找。(譯者注:即在 __mro__ 中的下一個(gè)基類中查找)

注意:在Python 2.2中,如果 m 是一個(gè)描述器, super(B, obj).m() 只會(huì)調(diào)用方法 __get__() 。在Python 2.3中,非資料描述器(除非是個(gè)舊式類)也會(huì)被調(diào)用。 super_getattro() 的實(shí)現(xiàn)細(xì)節(jié)在: Objects/typeobject.c ,[del] 一個(gè)等價(jià)的Python實(shí)現(xiàn)在 Guido's Tutorial [/del] (譯者注:原文此句已刪除,保留供大家參考)。

以上展示了描述器的機(jī)理是在 object, type, 和 super 的 __getattribute__() 方法中實(shí)現(xiàn)的。由 object 派生出的類自動(dòng)的繼承這個(gè)機(jī)理,或者它們有個(gè)有類似機(jī)理的元類。同樣,可以重寫類的 __getattribute__() 方法來關(guān)閉這個(gè)類的描述器行為。

描述器例子
下面的代碼中定義了一個(gè)資料描述器,每次 get 和 set 都會(huì)打印一條消息。重寫 __getattribute__() 是另一個(gè)可以使所有屬性擁有這個(gè)行為的方法。但是,描述器在監(jiān)視特定屬性的時(shí)候是很有用的。

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.xRetrieving var "x"10>>> m.x = 20Updating var "x">>> m.xRetrieving var "x"20>>> m.y5

這個(gè)協(xié)議非常簡(jiǎn)單,并且提供了令人激動(dòng)的可能。一些用途實(shí)在是太普遍以致于它們被打包成獨(dú)立的函數(shù)。像屬性(property), 方法(bound和unbound method), 靜態(tài)方法和類方法都是基于描述器協(xié)議的。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 若尔盖县| 兴隆县| 莱芜市| 白朗县| 长春市| 桂阳县| 通州市| 克什克腾旗| 永登县| 洪湖市| 吉木乃县| 彭阳县| 柘荣县| 曲周县| 伊宁市| 曲周县| 兴海县| 麦盖提县| 乌拉特后旗| 伊川县| 磐石市| 丁青县| 蒲江县| 马公市| 佛山市| 临朐县| 威远县| 河南省| 广汉市| 哈尔滨市| 罗甸县| 苗栗市| 大名县| 同心县| 汉中市| 绩溪县| 广河县| 利辛县| 绥德县| 龙泉市| 中宁县|