解決python super()調(diào)用多重繼承函數(shù)的問(wèn)題
當(dāng)類間繼承關(guān)系很簡(jiǎn)單時(shí),super()的使用很簡(jiǎn)單。
class A(object): def __init__(self): print('a') class B(A): def __init__(self): super(B, self).__init__() print('b') b = B()
輸出結(jié)果:
a b
當(dāng)一個(gè)類繼承多個(gè)類時(shí),問(wèn)題就復(fù)雜起來(lái)了,請(qǐng)看下例:
class A(object): def __init__(self): print('a') class B(object): def __init__(self): print('b') class C(A, B): def __init__(self): super(C, self).__init__() print('c') c = C()
咋一看,情況好像也不復(fù)雜,結(jié)果輸出a, c嘛。沒(méi)錯(cuò)!但是如果C類想同時(shí)調(diào)用A與B的__init__()呢?
有童鞋就要說(shuō)了,我顯示調(diào)用不就OK了嘛?
class A(object): def __init__(self): print('a') class B(object): def __init__(self): print('b') class C(A, B): def __init__(self): A.__init__() B.__init__() print('c') c = C()
效果一樣,還不夠好。因?yàn)闆](méi)有調(diào)用super(),super的一大好處在于,當(dāng)父類的名字修改時(shí),其繼承類不用修改調(diào)用方法。
下面給出完美解決方案:
class A(object): def __init__(self): super(A, self).__init__() print('a') class B(object): def __init__(self): super(B, self).__init__() print('b') class C(A, B): def __init__(self): super(C, self).__init__() print('c') print(C.mro()) c = C()
print(C.mro()),在實(shí)際中可以去掉,為啥寫在這里,后面再說(shuō)。
輸出結(jié)果:
[<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>, <type 'object'>] b a c
注意:輸出結(jié)果是b, a, c 而非a, b, c。為什么?
這里就要用上面的mro()輸出來(lái)解釋了。MRO全稱Method Resolution Order, 就是用來(lái)定義繼承方法的調(diào)用順序,自Python2.3以來(lái),MRO采用廣度優(yōu)先(區(qū)別于深度優(yōu)先)的規(guī)則定義。按廣度優(yōu)先的規(guī)則,出來(lái)的順序就是:
[<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>, <type 'object'>]
而每次調(diào)用super()則是,調(diào)用MRO中下一個(gè)函數(shù)。上面的例子中:super(C, self)則指向MRO中的下一個(gè)類(A), 于是調(diào)用A的init --> 在A的init中,又調(diào)用了super(),于是調(diào)用MRO中的下一個(gè)函數(shù)(B) --> B調(diào)用下一個(gè)(object), object啥也不干 --> 返回B中,print('b') --> 返回A中,print('a') --> 返回C中,print('c')。
這里再次強(qiáng)調(diào)一次,super(type, obj).func()函數(shù)調(diào)用的是,obj實(shí)例在MRO中下一個(gè)父類的可調(diào)用func(),而不是type的父類中的func()(這個(gè)是本文第一個(gè)示例給你帶來(lái)的錯(cuò)覺(jué))。
以上這篇解決python super()調(diào)用多重繼承函數(shù)的問(wèn)題就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
python正則表達(dá)式及使用正則表達(dá)式的例子
正則表達(dá)是用來(lái)匹配字符串,這篇文章給大家介紹了python正則表達(dá)式及正則表達(dá)式的例子,文章給大家提到了正則表達(dá)式語(yǔ)法規(guī)則,感興趣的朋友一起看看吧2018-01-01GitHub 熱門:Python 算法大全,Star 超過(guò) 2 萬(wàn)
4 月 27 日,GitHub 趨勢(shì)榜第 3 位是一個(gè)用 Python 編碼實(shí)現(xiàn)的算法庫(kù),Star 數(shù)早已達(dá)到 26000+2019-04-04使用Python測(cè)試Ping主機(jī)IP和某端口是否開(kāi)放的實(shí)例
今天小編就為大家分享一篇使用Python測(cè)試Ping主機(jī)IP和某端口是否開(kāi)放的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-12-12python基于隱馬爾可夫模型實(shí)現(xiàn)中文拼音輸入
這篇文章主要介紹了python基于隱馬爾可夫模型實(shí)現(xiàn)中文拼音輸入的相關(guān)資料,需要的朋友可以參考下2016-04-04基于深度學(xué)習(xí)和OpenCV實(shí)現(xiàn)目標(biāo)檢測(cè)
這篇文章主要介紹了通過(guò)使用OpenCV進(jìn)行基于深度學(xué)習(xí)的對(duì)象檢測(cè)以及使用OpenCV檢測(cè)視頻,文中的示例代碼講解詳細(xì),需要的可以參考一下2021-12-12Python3標(biāo)準(zhǔn)庫(kù)之functools管理函數(shù)的工具詳解
functools模塊提供的主要工具就是partial類,可以用來(lái)“包裝”一個(gè)有默認(rèn)參數(shù)的callable對(duì)象。這篇文章主要介紹了Python3標(biāo)準(zhǔn)庫(kù)functools管理函數(shù)的工具的實(shí)例詳解,需要的朋友可以參考下2020-02-02使用python怎樣產(chǎn)生10個(gè)不同的隨機(jī)數(shù)
這篇文章主要介紹了使用python實(shí)現(xiàn)產(chǎn)生10個(gè)不同的隨機(jī)數(shù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07使用Atom支持基于Jupyter的Python開(kāi)教程詳解
這篇文章主要介紹了使用Atom支持基于Jupyter的Python開(kāi)發(fā),Vscode雖然說(shuō)也有連接Jupyter的工具,但是交互式的開(kāi)發(fā)Hydrogen體驗(yàn)更好,需要的朋友可以參考下2021-08-08Python實(shí)現(xiàn)服務(wù)端渲染SSR的示例代碼
服務(wù)端渲染是一種常見(jiàn)的技術(shù)策略,特別是在需要改善網(wǎng)站的搜索引擎優(yōu)化(SEO)和首屏加載時(shí)間的場(chǎng)景下,本文將介紹如何利用?Python?實(shí)現(xiàn)?SSR,感興趣的可以了解下2024-02-02