繼承是面向對象編程的一個重要的方式,經過繼承,子類就能夠擴展父類的功能。在python中一個類能繼承自不止一個父類,這叫作python的多重繼承(Multiple Inheritance )。html
語法python
class SubclassName(BaseClass1, BaseClass2, BaseClass3, ...): pass
在多層繼承和多繼承同時使用的狀況下,就會出現複雜的繼承關係,多重多繼承。算法
其中,就會出現菱形繼承。以下圖所示。編程
在這種結構中,在調用順序上就出現了疑惑,調用順序到底是如下哪種順序呢微信
下面咱們來解答下這個問題。spa
舉個例子來看下:code
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()
輸出結果htm
init D... init B... init A... end A... end B... init C... init A... end A... end C... end D...
從輸出結果中看,調用順序爲:D->B->A->C->A。能夠看到,B、C共同繼承於A,A被調用了兩次。A不必重複調用兩次。對象
其實,上面問題的根源都跟MRO有關,MRO(Method Resolution Order)也叫方法解析順序,主要用於在多重繼承時判斷調的屬性來自於哪一個類,其使用了一種叫作C3的算法,其基本思想時在避免同一類被調用屢次的前提下,使用廣度優先和從左到右的原則去尋找須要的屬性和方法。blog
那麼如何避免頂層父類中的某個方法被屢次調用呢,此時就須要super()來發揮做用了,super本質上是一個類,內部記錄着MRO信息,因爲C3算法確保同一個類只會被搜尋一次,這樣就避免了頂層父類中的方法被屢次執行了,上面代碼能夠改成:
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()
輸出結果:
init D... init B... init C... init A... end A... end C... end B... end D...
能夠看出,此時的調用順序是D->B->C->A。即採用是廣度優先的遍歷方式。
Python類分爲兩種,一種叫經典類,一種叫新式類。都支持多繼承,但繼承順序不一樣。
Python2.x中類的是有經典類和新式類兩種。Python3.x中都是新式類。
打開微信掃一掃,關注【西加加先生】微信公衆號,及時接收博文推送。
原文出處:https://www.cnblogs.com/ghostlee/p/12298294.html