這里的官方文檔描述了在形如a.b的過程中都發(fā)生了什么:
首先會嘗試a.__dict__['b'],然后如果沒找到,則type(a).__dict__['b'],再沒找到則順著mro順序逐個(gè)嘗試,直到找到未知。且在找到之后查看這里的b是否定義有描述器相關(guān)的方法__get__ / __set__,如果有的話,則用描述器相關(guān)方法來取代默認(rèn)行為。
有下面這樣一個(gè)簡單例子:
class Descri(object):
def __get__(self, obj, type=None):
print("call get")
def __set__(self, obj, value):
print("call set")
class A(object):
x = Descri()
a = A()
a.__dict__['x'] = 1
在最后一行執(zhí)行完畢之后,我們獲取a.x。
按照前面說的查找順序,應(yīng)該直接在實(shí)例a的__dict__中就能夠找到x屬性,且其等于1并且沒有定義任何的特殊方法,那么按照查找優(yōu)先級來看應(yīng)該直接返回1,而且不應(yīng)該繼續(xù)查找任何的type(a).__dict__才對。但是實(shí)際卻不是這樣,實(shí)際上依然調(diào)用了Descri.__get__方法。
請問這是為什么?
先說下描述符的分類:
再說下描述符的訪問優(yōu)先級:
類.屬性 > 數(shù)據(jù)描述符 > 對象.屬性 > 非數(shù)據(jù)描述符
當(dāng)你用a.__dict__['x'] = 1,會向a的名稱空間寫入{'x': 1}。但是當(dāng)你用a.x訪問時(shí),由于你實(shí)現(xiàn)的是數(shù)據(jù)描述符,優(yōu)先級高于對象.屬性,此時(shí)會執(zhí)行type(a).__dict__['x'].__get__(a,type(a)),而不是a.__dict__['x']。
如果你希望執(zhí)行a.x時(shí),返回a.__dict__['x']。把上面的__set__注釋掉就好了。因?yàn)榇藭r(shí)實(shí)現(xiàn)的是非數(shù)據(jù)描述法,優(yōu)先級小于對象.屬性。
希望對你有幫助!
謝邀,a.x 在實(shí)例屬性的變量空間就命中了,因此不會到上級的類的變量空間再查找了。在我這的測試,也并沒有調(diào)用 Descri.__get__:
本地的 3.6 版本也是一樣的結(jié)果。
======分割線=====
根據(jù)題主說的在最后一行試試 a.x ,還真是執(zhí)行了 Descri.__get__ 。這種詭異的調(diào)用順序其實(shí)是在文檔里就有的解釋的。
一個(gè)屬性的訪問已被描述符協(xié)議方法重寫對象屬性: __get__(),__set__(),和 __delete__()。如果為對象定義了任何這些方法,則稱其為描述符。
一般情況下(我是說一般情況下),a.x 的查找鏈?zhǔn)牵?a.__dict__['x'] 然后再是 type(a).__dict__['x'] ,若沒找到則繼續(xù)通過 type(a) 的基類查找。
但是但是,如果查找的值其中一個(gè)是定義了描述符方法的對象,那么 Python 會覆蓋這個(gè)默認(rèn)行為轉(zhuǎn)而調(diào)用描述符的方法。這種行為也會因?yàn)檎{(diào)用的不同而稍有不一樣:
a.x 則轉(zhuǎn)換為調(diào)用: 。type(a).__dict__['x'].__get__(a, type(a))
A.x 則轉(zhuǎn)換為:A.__dict__['x'].__get__(None, A)
更多參考可見文檔:https://docs.python.org/3/ref...
北大青鳥APTECH成立于1999年。依托北京大學(xué)優(yōu)質(zhì)雄厚的教育資源和背景,秉承“教育改變生活”的發(fā)展理念,致力于培養(yǎng)中國IT技能型緊缺人才,是大數(shù)據(jù)專業(yè)的國家
達(dá)內(nèi)教育集團(tuán)成立于2002年,是一家由留學(xué)海歸創(chuàng)辦的高端職業(yè)教育培訓(xùn)機(jī)構(gòu),是中國一站式人才培養(yǎng)平臺、一站式人才輸送平臺。2014年4月3日在美國成功上市,融資1
北大課工場是北京大學(xué)校辦產(chǎn)業(yè)為響應(yīng)國家深化產(chǎn)教融合/校企合作的政策,積極推進(jìn)“中國制造2025”,實(shí)現(xiàn)中華民族偉大復(fù)興的升級產(chǎn)業(yè)鏈。利用北京大學(xué)優(yōu)質(zhì)教育資源及背
博為峰,中國職業(yè)人才培訓(xùn)領(lǐng)域的先行者
曾工作于聯(lián)想擔(dān)任系統(tǒng)開發(fā)工程師,曾在博彥科技股份有限公司擔(dān)任項(xiàng)目經(jīng)理從事移動(dòng)互聯(lián)網(wǎng)管理及研發(fā)工作,曾創(chuàng)辦藍(lán)懿科技有限責(zé)任公司從事總經(jīng)理職務(wù)負(fù)責(zé)iOS教學(xué)及管理工作。
浪潮集團(tuán)項(xiàng)目經(jīng)理。精通Java與.NET 技術(shù), 熟練的跨平臺面向?qū)ο箝_發(fā)經(jīng)驗(yàn),技術(shù)功底深厚。 授課風(fēng)格 授課風(fēng)格清新自然、條理清晰、主次分明、重點(diǎn)難點(diǎn)突出、引人入勝。
精通HTML5和CSS3;Javascript及主流js庫,具有快速界面開發(fā)的能力,對瀏覽器兼容性、前端性能優(yōu)化等有深入理解。精通網(wǎng)頁制作和網(wǎng)頁游戲開發(fā)。
具有10 年的Java 企業(yè)應(yīng)用開發(fā)經(jīng)驗(yàn)。曾經(jīng)歷任德國Software AG 技術(shù)顧問,美國Dachieve 系統(tǒng)架構(gòu)師,美國AngelEngineers Inc. 系統(tǒng)架構(gòu)師。