国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術文章
文章詳情頁

關于python的繼承的一個疑問

瀏覽:200日期:2022-08-18 14:17:38

問題描述

關于python的繼承的一個疑問

如圖所示,B 類繼承了 A 類;

當實例化對象時,B 類中沒有構造函數, 應該調用父類的構造函數 __init__

但是里邊的 self.__pirv() 為啥調用到父類 A 的 __priv, 而 self.pub() 又調到 B 中的 pub

求解?

問題解答

回答1:

在Python從__開始的方法名稱不是private,__的意思就讓Python做name mangling,name mangling的結果就是_A__priv。這樣的方法你應該不要overwrite。如果你想讓子類overwrite一個方法你只用_,不用__

回答2:

想了一下這個問題,我是這樣理解的:

class A(object): def __init__(self):self.__priv() # _A__priv()self.pub() def __priv(self):print('private A') def pub(self):print('public A')class B(A): def __init__(self):self.__priv() # 調用_B__priv()對比super(B, self).__init__() # 在這里顯式調用父類的`__init__()`方法 def __priv(self):print('private B') def pub(self):print('public B')if __name__ == ’__main__’: b = B()

在子類的實例調用__init__()方法時,從子類B本身中查找pub()方法,顯然是存在的,因此會調用B類自身的pub()方法;然而在調用__priv()方法時,由于Python對私有成員進行了名稱改編,你實際調用的是_A__priv()方法,而B類中并沒有這個方法,有的只有_B__priv(),因此調用父類A中的_A__priv(),于是產生了這樣的結果。這是我個人的理解,如果有誤歡迎指正,謝謝。

回答3:

關於問題本身, @Xavier 和 @Christoph 已經有詳細的說明了

如果你還不明白可以試著這樣做:

原本的代碼:

class A: def __init__(self):self.__priv() # 等等改成 self._A__priv()self.public() def __priv(self): # 等等改成 def _A__priv(self):print(’private of A’) def public(self):print(’public of A’)class B(A): def __priv(self): # 等等改成 self._B__priv(self):print(’private of B’) def public(self):print(’public of B’)b = B()

自行手動進行 name mangling:

class A: def __init__(self):self._A__priv()self.public() def _A__priv(self):print(’private of A’) def public(self):print(’public of A’)class B(A): def _B__priv(self):print(’private of B’) def public(self):print(’public of B’)b = B()

B 在這裡繼承了 A 的所有屬性包含:

__init__

_A__priv

public

而 B 自己定義了:

_B__priv

public (此處覆寫了 A 的 public)

所以最後你會看到 dir(b) 裡面有:

__init__ (從 A 繼承的)

_A__priv (從 A 繼承的)

_B__priv (自己定義的)

public (自己定義的)

最後當 __init__ 被呼叫時, 會調用 _A__priv, 而 B 裡面的確有這個方法

囉唆補充一下, Python 本身並沒有真正的 private 機制, 因為了解 name mangling 的人就能對以雙底線開頭的屬性做存取, 比如說我可以很輕易地寫出:

a = A()a._A__priv() # 防都防不住

簡單來說這個機制是個:

防呆 的機制, 不是個 防小人 的機制,

防止意外 存取的機制, 不是個防止 刻意存取 的機制

但是這個機制並非所有人都覺得好(個人就不喜歡, 使用雙底線開頭命名既麻煩也沒太多實際的幫助), 所以你可以在很多的 python 代碼中發現: 大家比較常使用以 單個底線 開頭的保護方式, 這種做法是個公認的慣例(註1), 對於稍有經驗的程序員來說足以防呆, 且不會有任何額外的效果和意外的狀況發生

Ian Bicking 有一段話是這樣說的 (Ian Bicking 是 Python 大神, 這段話我是在 Luciano Ramalho 的的 Fluent Python 中看到的):

永遠不要在前面使用兩個底線. 這是很讓人生氣的自私行為, 如果你不希望造成名稱衝突(註2), 可以明確地重整名稱(例如: _MyThing_blahblah). 實質上這與使用雙底線是同一件事情, 不過他是公開的, 雙底線是私下的行為.

所以我的建議是, 使用 _priv 會是更好的選擇.

註1: 以單底線開頭的屬性不會具有任何特殊的性質, 他僅僅是依靠 Python 程序員的共識而產生的具有象徵意義的符號性手法, 就好像有些語言會使用 const 來標明常量, 而我們也可以僅依賴 常量使用大寫命名 的共識來避免意外的狀況發生

註2: 之所以想要以 private 性質來保護屬性, 最常見的就是因為名稱衝突引起的意外存取

我回答過的問題: Python-QA

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 午夜爽爽爽 | 精品伊人久久久久7777人 | 在线观看国产精成人品 | 国外成人在线视频 | 亚洲国产高清视频 | 亚洲精品国产一区二区三 | 精品欧美一区二区三区精品久久 | 99re5久久在热线播放 | 美女张开腿让人桶 | 高跟丝袜美女一级毛片 | 美国毛片aa | 成人精品视频一区二区在线 | 三级特黄视频 | 香蕉521av网站永久地址 | 精品国产三级a∨在线观看 精品国产三级a在线观看 | 欧美在线成人午夜网站 | 一级毛片 在线播放 | 精品视频一区二区三区四区 | 国产三级做爰高清在线 | 亚洲欧美综合国产不卡 | 大尺度福利视频奶水在线 | 国产亚洲欧洲一区二区三区 | 久久999精品| 99爱在线视频 | 欧美丰满大乳大屁股毛片 | 中文字幕福利片 | 日本三级11k影院在线 | 成人性一级视频在线观看 | 精品国产网 | 国产v综合v亚洲欧美大另类 | 瑟瑟网站在线观看 | 国产午夜精品理论片久久影视 | 台湾精品视频在线播放 | 99精品视频观看 | 可以看毛片的网站 | 一级做a爰片久久毛片看看 一级做a爰片久久毛片鸭王 | 亚洲欧美字幕 | 欧美综合自拍亚洲综合百度 | 欧美成人手机视频 | 久久精品夜色国产 | 久艹在线视频 |