Python多重繼承之菱形繼承的實(shí)例詳解
繼承是面向?qū)ο缶幊痰囊粋€(gè)重要的方式,通過(guò)繼承,子類(lèi)就可以擴(kuò)展父類(lèi)的功能。在python中一個(gè)類(lèi)能繼承自不止一個(gè)父類(lèi),這叫做python的多重繼承(Multiple Inheritance )。
語(yǔ)法
class SubclassName(BaseClass1, BaseClass2, BaseClass3, ...): pass
菱形繼承
在多層繼承和多繼承同時(shí)使用的情況下,就會(huì)出現(xiàn)復(fù)雜的繼承關(guān)系,多重多繼承。
其中,就會(huì)出現(xiàn)菱形繼承。如下圖所示。mark
在這種結(jié)構(gòu)中,在調(diào)用順序上就出現(xiàn)了疑惑,調(diào)用順序究竟是以下哪一種順序呢
- D->B->A->C(深度優(yōu)先)
- D->B->C->A(廣度優(yōu)先)
下面我們來(lái)解答下這個(gè)問(wèn)題。
舉個(gè)例子來(lái)看下:
class A(): def __init__(self): print('init A...') print('end A...') class B(A): def __init__(self): print('init B...') A.__init__(self) print('end B...') class C(A): def __init__(self): print('init C...') A.__init__(self) print('end C...') class D(B, C): def __init__(self): print('init D...') B.__init__(self) C.__init__(self) print('end D...') if __name__ == '__main__': D()
輸出結(jié)果
init D... init B... init A... end A... end B... init C... init A... end A... end C... end D...
從輸出結(jié)果中看,調(diào)用順序?yàn)椋篋->B->A->C->A??梢钥吹?,B、C共同繼承于A,A被調(diào)用了兩次。A沒(méi)必要重復(fù)調(diào)用兩次。
其實(shí),上面問(wèn)題的根源都跟MRO有關(guān),MRO(Method Resolution Order)也叫方法解析順序,主要用于在多重繼承時(shí)判斷調(diào)的屬性來(lái)自于哪個(gè)類(lèi),其使用了一種叫做C3的算法,其基本思想時(shí)在避免同一類(lèi)被調(diào)用多次的前提下,使用廣度優(yōu)先和從左到右的原則去尋找需要的屬性和方法。
那么如何避免頂層父類(lèi)中的某個(gè)方法被多次調(diào)用呢,此時(shí)就需要super()來(lái)發(fā)揮作用了,super本質(zhì)上是一個(gè)類(lèi),內(nèi)部記錄著MRO信息,由于C3算法確保同一個(gè)類(lèi)只會(huì)被搜尋一次,這樣就避免了頂層父類(lèi)中的方法被多次執(zhí)行了,上面代碼可以改為:
class A(): def __init__(self): print('init A...') print('end A...') class B(A): def __init__(self): print('init B...') super(B, self).__init__() print('end B...') class C(A): def __init__(self): print('init C...') super(C, self).__init__() print('end C...') class D(B, C): def __init__(self): print('init D...') super(D, self).__init__() print('end D...') if __name__ == '__main__': D()
輸出結(jié)果:
init D... init B... init C... init A... end A... end C... end B... end D...
可以看出,此時(shí)的調(diào)用順序是D->B->C->A。即采用是廣度優(yōu)先的遍歷方式。
補(bǔ)充內(nèi)容
Python類(lèi)分為兩種,一種叫經(jīng)典類(lèi),一種叫新式類(lèi)。都支持多繼承,但繼承順序不同。
新式類(lèi):從object繼承來(lái)的類(lèi)。(如:class A(object)),采用廣度優(yōu)先搜索的方式繼承(即先水平搜索,再向上搜索)。
經(jīng)典類(lèi):不從object繼承來(lái)的類(lèi)。(如:class A()),采用深度優(yōu)先搜索的方式繼承(即先深入繼承樹(shù)的左側(cè),再返回,再找右側(cè))。
Python2.x中類(lèi)的是有經(jīng)典類(lèi)和新式類(lèi)兩種。Python3.x中都是新式類(lèi)。
總結(jié)
以上所述是小編給大家介紹的Python多重繼承之菱形繼承的實(shí)例詳解,希望對(duì)大家有所幫助!
相關(guān)文章
Python的Scrapy爬蟲(chóng)框架簡(jiǎn)單學(xué)習(xí)筆記
這篇文章主要介紹了Python的Scrapy爬蟲(chóng)框架簡(jiǎn)單學(xué)習(xí)筆記,從基本的創(chuàng)建項(xiàng)目到CrawlSpider的使用等都有涉及,需要的朋友可以參考下2016-01-01Numpy中Meshgrid函數(shù)基本用法及2種應(yīng)用場(chǎng)景
NumPy包含很多實(shí)用的數(shù)學(xué)函數(shù),涵蓋線(xiàn)性代數(shù)運(yùn)算、傅里葉變換和隨機(jī)數(shù)生成等功能,下面這篇文章主要給大家介紹了關(guān)于Numpy中Meshgrid函數(shù)基本用法及2種應(yīng)用場(chǎng)景的相關(guān)資料,需要的朋友可以參考下2022-08-08PYQT5 vscode聯(lián)合操作qtdesigner的方法
這篇文章主要介紹了PYQT5 vscode聯(lián)合操作qtdesigner的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03Python數(shù)據(jù)可視化之Matplotlib初級(jí)使用指南
Matplotlib是Python中最常用的數(shù)據(jù)可視化庫(kù)之一,它提供了豐富的圖表類(lèi)型和靈活的自定義選項(xiàng),能幫助我們以更直觀的方式理解數(shù)據(jù),本文將對(duì)Matplotlib的基本功能進(jìn)行介紹,包括如何創(chuàng)建和自定義圖表等2023-07-07Python實(shí)現(xiàn)去除圖片中指定顏色的像素功能示例
這篇文章主要介紹了Python實(shí)現(xiàn)去除圖片中指定顏色的像素功能,結(jié)合具體實(shí)例形式分析了Python基于pil與cv2模塊的圖形載入、運(yùn)算、轉(zhuǎn)換等相關(guān)操作技巧,需要的朋友可以參考下2019-04-04解決安裝Python的第三方庫(kù)pandas報(bào)錯(cuò)問(wèn)題
這篇文章主要介紹了解決安裝Python的第三方庫(kù)pandas報(bào)錯(cuò)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08python卸載numpy出現(xiàn)WinError:拒絕訪(fǎng)問(wèn)的解決方案
這篇文章主要介紹了python卸載numpy出現(xiàn)WinError:拒絕訪(fǎng)問(wèn)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08