Python編程中對(duì)super函數(shù)的正確理解和用法解析
當(dāng)在子類需要調(diào)用父類的方法時(shí),在python2.2之前,直接用類名調(diào)用類的方法,即非綁定的類方法,并把自身對(duì)象self作參數(shù)傳進(jìn)去。
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
這樣運(yùn)作挺好,不過有個(gè)問題,當(dāng)父類改了名字時(shí),就要把這些顯式調(diào)用父類的一個(gè)個(gè)更正,子類和父類耦合比較高。
于是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在多重繼承有問題,如果子類繼承多個(gè)父類,那么super調(diào)用第一個(gè)父類的方法。
- 不要混用這兩種調(diào)用父類方法的方案,要么都用非綁定的類方法,要么都用super。不然可能導(dǎo)致沒被調(diào)用或者被調(diào)用多次。
BUT:
不要一說到 super 就想到父類!super 指的是 MRO 中的下一個(gè)類!
一說到 super 就想到父類這是初學(xué)者很容易犯的一個(gè)錯(cuò)誤,也是我當(dāng)年犯的錯(cuò)誤。
def super(cls, inst): mro = inst.__class__.mro() return mro[mro.index(cls) + 1]
兩個(gè)參數(shù) cls 和 inst 分別做了兩件事:
1. inst 負(fù)責(zé)生成 MRO 的 list
2. 通過 cls 定位當(dāng)前 MRO 中的 index, 并返回 mro[index + 1]
這兩件事才是 super 的實(shí)質(zhì),一定要記?。?br />
MRO 全稱 Method Resolution Order,它代表了類繼承的順序。
舉個(gè)例子:
class Root(object):
def __init__(self):
print("this is Root")
class B(Root):
def __init__(self):
print("enter B")
# print(self) # this will print <__main__.D object at 0x...>
super(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 (<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.Root'>, <type 'object'>)
知道了 super 和父類其實(shí)沒有實(shí)質(zhì)關(guān)聯(lián)之后,我們就不難理解為什么 enter B 下一句是 enter C 而不是 this is Root(如果認(rèn)為 super 代表“調(diào)用父類的方法”,會(huì)想當(dāng)然的認(rèn)為下一句應(yīng)該是this is Root)。流程如下,在 B 的 __init__ 函數(shù)中:
super(B, self).__init__()
首先,我們獲取 self.__class__.__mro__,注意這里的 self 是 D 的 instance 而不是 B 的
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.Root'>, <type 'object'>)
順便說一句為什么 B 的 __init__ 會(huì)被調(diào)用:因?yàn)?D 沒有定義 __init__,所以會(huì)在 MRO 中找下一個(gè)類,去查看它有沒有定義 __init__,也就是去調(diào)用 B 的 __init__。
其實(shí)這一切邏輯還是很清晰的,關(guān)鍵是理解 super 到底做了什么。
相關(guān)文章
python使用循環(huán)打印所有三位數(shù)水仙花數(shù)的實(shí)例
今天小編就為大家分享一篇python使用循環(huán)打印所有三位數(shù)水仙花數(shù)的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-11-11
Python+?Flask實(shí)現(xiàn)Mock?Server詳情
這篇文章主要介紹了Python+?Flask實(shí)現(xiàn)Mock?Server詳情,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-09-09
Python3安裝模塊報(bào)錯(cuò)Microsoft Visual C++ 14.0 is required的解決方法
這篇文章主要介紹了Python3安裝模塊報(bào)錯(cuò)Microsoft Visual C++ 14.0 is required的解決方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
使用Python機(jī)器學(xué)習(xí)降低靜態(tài)日志噪聲
今天小編就為大家分享一篇關(guān)于使用Python和機(jī)器學(xué)習(xí)的靜態(tài)日志噪聲的文章,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2018-09-09
opencv+python實(shí)現(xiàn)鼠標(biāo)點(diǎn)擊圖像,輸出該點(diǎn)的RGB和HSV值
這篇文章主要介紹了opencv+python實(shí)現(xiàn)鼠標(biāo)點(diǎn)擊圖像,輸出該點(diǎn)的RGB和HSV值,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-06-06
Python讀寫文件基礎(chǔ)知識(shí)點(diǎn)
在本篇文章中小編給大家整理了關(guān)于Python讀寫文件的基礎(chǔ)知識(shí)內(nèi)容,有興趣的朋友們跟著學(xué)習(xí)下。2019-06-06
Python 生成 -1~1 之間的隨機(jī)數(shù)矩陣方法
今天小編就為大家分享一篇Python 生成 -1~1 之間的隨機(jī)數(shù)矩陣方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-08-08
python中os.path.dirname(path)詳細(xì)解釋和使用示例
這篇文章主要介紹了python中os.path.dirname(path)詳細(xì)解釋和使用示例,os.path.dirname是一個(gè)Python函數(shù),用于獲取文件路徑的目錄部分,它通常與os.path.basename結(jié)合使用,以分離路徑中的目錄和文件名,需要的朋友可以參考下2025-03-03

