class A(object): def say(self): print 'I am A' class B(A): def say(self): print 'I am B' A.say(self) b = B() b.say()
輸出
I am B I am A
這樣運作挺好,不過有個問題,當(dāng)父類改了名字時,就要把這些顯式調(diào)用父類的一個個更正,子類和父類耦合比較高。
于是python2.2后就推出了super()函數(shù)來避免硬編碼,不用關(guān)心父類名叫什么。
使用super()函數(shù),上面的代碼可以寫成如下。
class B(A): def say(self): print 'I am B' super(B,self).say()
python3.0后,又做了改良,super()函數(shù)不用傳參數(shù),即上面的那行代碼直接super().say()就行了。
需要注意的問題:
super只能用在新式類中。
super在多重繼承有問題,如果子類繼承多個父類,那么super調(diào)用第一個父類的方法。
不要混用這兩種調(diào)用父類方法的方案,要么都用非綁定的類方法,要么都用super。不然可能導(dǎo)致沒被調(diào)用或者被調(diào)用多次。
BUT:
不要一說到 super 就想到父類!super 指的是 MRO 中的下一個類!
一說到 super 就想到父類這是初學(xué)者很容易犯的一個錯誤,也是我當(dāng)年犯的錯誤。
def super(cls, inst): mro = inst.__class__.mro() return mro[mro.index(cls) + 1]
兩個參數(shù) cls 和 inst 分別做了兩件事:
1. inst 負(fù)責(zé)生成 MRO 的 list
2. 通過 cls 定位當(dāng)前 MRO 中的 index, 并返回 mro[index + 1]
這兩件事才是 super 的實質(zhì),一定要記住!
MRO 全稱 Method Resolution Order,它代表了類繼承的順序。
舉個例子:
class Root(object): def __init__(self): print("this is Root") class B(Root): def __init__(self): print("enter B") # print(self) # this will printsuper(B, self).__init__() print("leave B") class C(Root): def __init__(self): print("enter C") super(C, self).__init__() print("leave C") class D(B, C): pass d = D() print(d.__class__.__mro__)
輸出
enter B enter C this is Root leave C leave B (,,,,)
知道了 super 和父類其實沒有實質(zhì)關(guān)聯(lián)之后,我們就不難理解為什么 enter B 下一句是 enter C 而不是 this is Root(如果認(rèn)為 super 代表“調(diào)用父類的方法”,會想當(dāng)然的認(rèn)為下一句應(yīng)該是this is Root)。流程如下,在 B 的 __init__ 函數(shù)中:
super(B, self).__init__()
首先,我們獲取 self.__class__.__mro__,注意這里的 self 是 D 的 instance 而不是 B 的
(
然后,通過 B 來定位 MRO 中的 index,并找到下一個。顯然 B 的下一個是 C。于是,我們調(diào)用 C 的 __init__,打出 enter C。
順便說一句為什么 B 的 __init__ 會被調(diào)用:因為 D 沒有定義 __init__,所以會在 MRO 中找下一個類,去查看它有沒有定義 __init__,也就是去調(diào)用 B 的 __init__。
其實這一切邏輯還是很清晰的,關(guān)鍵是理解 super 到底做了什么。
聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com