python?super()函數(shù)的詳解
Python是一門面向?qū)ο蟮恼Z言,定義類時(shí)經(jīng)常要用到繼承,在類的繼承中,子類繼承父類中已經(jīng)封裝好的方法,不需要再次編寫,如果子類如果重新定義了父類的某一方法,那么該方法就會(huì)覆蓋父類的同名方法,但是有時(shí)我們希望子類保持父類方法的基礎(chǔ)上進(jìn)行擴(kuò)展,而不是直接覆蓋,就需要先調(diào)用父類的方法,然后再進(jìn)行功能的擴(kuò)展,這時(shí)就可以通過super來實(shí)現(xiàn)對(duì)父類方法的調(diào)用。
super的用法
看下面一個(gè)例子:
class A: def func(self): print("A的func執(zhí)行") class B(A): def func(self): super().func() print("B擴(kuò)展的func執(zhí)行") b = B() b.func()
# 輸出結(jié)果為:
# A的func執(zhí)行
# B擴(kuò)展的func執(zhí)行
上面程序中,A是父類,B是A的子類,我們?cè)贏類中重定義了func()方法,在B類中重新定義了func()方法,在方法中通過super().func()又調(diào)用了父類的方法,所以執(zhí)行結(jié)果才會(huì)有A類func()方法輸出。
如果經(jīng)常看Python內(nèi)置庫及第三方庫源碼的話,你會(huì)發(fā)現(xiàn),super用的非常多的地方是在子類中調(diào)用父類的初始化__init__()方法,這種用法非常常見。
class A: def __init__(self, x): self.x = x class B(A): def __init__(self, x, y): super().__init__(x) self.y = y b = B(1, 2) print(b.x, b.y)
看到這,你會(huì)想到super就是用來獲取父類并用來調(diào)用父類方法的,這樣說對(duì)不對(duì)呢,其實(shí)是不對(duì)的,使用supper獲取的不是父類,而是MRO列表中的下一個(gè)類,所謂MRO列表即方法解析順序(Method Resolution Order)列表,它代表著類繼承的順序,我們可以使用以下幾種獲得某個(gè)類的MRO列表:
C.mro() C.__mro__ c.__class__.__mro__
MRO列表的順序確定經(jīng)歷了很多次的變遷,最新的是通過C3線性化算法來實(shí)現(xiàn)的,感興趣的話可以自行了解一下,總的來說,一個(gè)類的MRO列表就是合并所有父類的MRO列表,并遵循以下三條原則:
- 子類永遠(yuǎn)在父類前面
- 如果有多個(gè)父類,會(huì)根據(jù)它們?cè)诹斜碇械捻樞虮粰z查
- 如果對(duì)下一個(gè)類存在兩個(gè)合法的選擇,選擇第一個(gè)父類
下面來看一下下面這個(gè)例子:
class A(Base): def func(self): print("A的func執(zhí)行") super().func() print("A的func執(zhí)行完畢") class B(Base): def func(self): print("B的func執(zhí)行") super().func() print("B的func執(zhí)行完畢") class C(A, B): def func(self): print("C的func執(zhí)行") super().func() print("C的func執(zhí)行完畢") c = C() c.func() # 獲取MRO列表 print(c.__class__.__mro__)
執(zhí)行結(jié)果如下:
上述程序中,Base是父類,A、B都繼承自Base,C繼承自 A、B,它們的繼承關(guān)系就是一個(gè)典型的菱形繼承,如下:
通過結(jié)果我們可以看出,super并不是獲取父類并用來調(diào)用父類的方法,而是根據(jù)MRO列表一次調(diào)用下一個(gè)類,使用c.__class__.__mro__可以獲取MRO列表,MRO列表的順序是C、A、B、Base、object。
super的原理
super計(jì)算方法解析順序中的下一個(gè)類,可以接收兩個(gè)參數(shù):
def super(cls, inst): mro = inst.__class__.mro() return mro[mro.index(cls) + 1]
總結(jié)
現(xiàn)在我們知道:supper獲取的是MRO列表中的下一個(gè)類,當(dāng)前類的父類沒有實(shí)質(zhì)性的關(guān)系;還有如何查看MRO列表。最后需要注意的是super以及MRO列表,針對(duì)都是Python新式類!
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
python嵌套字典比較值與取值的實(shí)現(xiàn)示例
這篇文章主要給大家介紹了關(guān)于python嵌套字典比較值與取值的實(shí)現(xiàn)方法,詳細(xì)介紹了python字典嵌套字典的情況下獲取某個(gè)key的value的相關(guān)內(nèi)容,分享出來供大家參考學(xué)習(xí),需要的朋友們下面來一起看看吧。2017-11-11Python實(shí)現(xiàn)掃描指定目錄下的子目錄及文件的方法
這篇文章主要介紹了Python實(shí)現(xiàn)掃描指定目錄下的子目錄及文件的方法,需要的朋友可以參考下2014-07-07Python多進(jìn)程Process和管道Pipe的使用方式
這篇文章主要介紹了Python多進(jìn)程Process和管道Pipe的使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02python+Tesseract OCR實(shí)現(xiàn)截屏識(shí)別文字
pytesseract Python常用pytesseract進(jìn)行圖片上的文字識(shí)別,本文主要介紹了python+Tesseract?OCR實(shí)現(xiàn)截屏識(shí)別文字,具有一定的參考價(jià)值,感興趣的可以了解一下2023-11-11python基于opencv實(shí)現(xiàn)人臉識(shí)別
這篇文章主要介紹了python基于opencv實(shí)現(xiàn)人臉識(shí)別的方法,幫助大家更好的理解和使用python,感興趣的朋友可以了解下2021-01-01