python?中的?super詳解
提到 super,最直接的想法就是它代表了父類,替父類執(zhí)行某些方法。但是理解也僅止步于此,下面對(duì) super 做進(jìn)一步理解
super 的完整形式
常見的 super 用法如下
class Person():
def __init__(self,name):
self.name = name
print('Person')
class Male(Person):
def __init__(self,age):
super().__init__('xiaoming')
self.age = age
print("Male")
m = Male(12)
print(m.__dict__)
以上執(zhí)行結(jié)果為

這個(gè)結(jié)果也符合理解,Male 繼承了 Person,在初始化的時(shí)候執(zhí)行了父類的初始化方法,也就繼承了父類的 name 屬性。
但是其實(shí) super 的完整形式為
super(Male, self).__init__('xiaoming')super 是一個(gè)類,其中第二個(gè)參數(shù)是個(gè) class 或者 object,決定了使用怎樣的 mro。第一個(gè)參數(shù)是個(gè) class,決定了從 mro 哪個(gè) class 后面的 class 開始尋找,并將函數(shù)綁定到第二個(gè)參數(shù)上。兩個(gè)參數(shù)都是可選的。
本例中,self 就是 Male 的實(shí)例對(duì)象,于是 self 的 mro 就是 [Male,Person,Object],而第一個(gè)參數(shù)是 Male,于是就使用 Male 后面的 Person,發(fā)現(xiàn) Person 有 __init__ 函數(shù),于是就只執(zhí)行 Person 的 __init__ 函數(shù),也就是 super 行的語句等價(jià)于
# super(Male, self).__init__('xiaoming')
Person.__init__(self,'xiaoming')
執(zhí)行結(jié)果同上

super 的使用
super 可以在定義類之外的地方使用
class Animal():
def __init__(self,name):
self.name = name
class Person(Animal):
def __init__(self,name,age):
super().__init__(name)
self.age = age
print('Person')
class Male(Person):
def __init__(self,name,age):
super(Person,self).__init__(name,age)
print("Male")
m = Male('xiaoming',12)
super(Male,m).__init__('xiaoming',12)
print(m.__dict__)
執(zhí)行結(jié)果為

可以看到 16 行報(bào)錯(cuò)了,報(bào)錯(cuò)的原因就是此時(shí)的 self 代表的是 Male 實(shí)例,Male 的 mro 是 Male,Person,Animal,Object。Male 在實(shí)例化的時(shí)候執(zhí)行了父類的 __init__ 方法,而此時(shí) super 的第一個(gè)參數(shù)是 Person,于是使用 Person 后面的 Animal,而 Animal 的 __init__ 方法只有一個(gè)參數(shù),super 卻傳遞了2個(gè)參數(shù),于是報(bào)錯(cuò)了。正確地修改為
# class Person: super(Person,self).__init__(name)
執(zhí)行結(jié)果為

可以看到 Male 實(shí)例化的時(shí)候繞過了 Person,只輸出了 Animal 和 Male。而在類之外執(zhí)行的 super,執(zhí)行了 Male 的父類(Person、Animal)的 __init__ 方法。 說明了 2 點(diǎn):
super的第一個(gè)參數(shù)決定了選擇self的 mro 哪個(gè) class 之后的 class。super可以在類定義之外執(zhí)行。
再看一個(gè)例子將會(huì)更加明白

直覺上來說,D 的實(shí)例會(huì)執(zhí)行父類的 say() ,首先會(huì)找到 B,于是會(huì)執(zhí)行 B 的父類的 say(),于是會(huì)輸出 'A'。結(jié)果卻是 'C',原因就是 self 代表了 D 的實(shí)例,而 D 的 mro 是 ['B','C','A'],D 的實(shí)例執(zhí)行父類的 say() ,會(huì)找到 B 執(zhí)行 B 的 super 方法,相當(dāng)于 super(B,self).say(),而此時(shí)的 self 代表 D,mro 搜索會(huì)選擇 B 后面的 class 也就是 C,執(zhí)行 C 的 say(),于是最終結(jié)果輸出 'C'
類中使用 super 的時(shí)候,可以省略參數(shù)而直接寫成 super(),這時(shí) super 會(huì)將他所在的類當(dāng)作第一個(gè)參數(shù),將所在函數(shù)的第一個(gè)參數(shù)當(dāng)作自己的第二個(gè)參數(shù)。顯然,這樣省略參數(shù)的 super 不能在類之外直接使用。
最后,查看一個(gè)類的 mro 可以用 class.__mro__ 或者 class.mro() 獲取

到此這篇關(guān)于python 中的 super的文章就介紹到這了,更多相關(guān)python super內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python 利用pandas將arff文件轉(zhuǎn)csv文件的方法
今天小編就為大家分享一篇python 利用pandas將arff文件轉(zhuǎn)csv文件的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-02-02
Python中Yield的基本用法及Yield與return的區(qū)別解析
Python中有一個(gè)非常有用的語法叫做生成器,用到的關(guān)鍵字就是yield,這篇文章主要介紹了Python中Yield的基本用法及Yield與return的區(qū)別,需要的朋友可以參考下2022-10-10
pandas進(jìn)行數(shù)據(jù)的交集與并集方式的數(shù)據(jù)合并方法
今天小編就為大家分享一篇pandas進(jìn)行數(shù)據(jù)的交集與并集方式的數(shù)據(jù)合并方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-06-06
10個(gè)使用Python必須知道的內(nèi)置函數(shù)
這篇文章小編主要向大家介紹的是10個(gè)使用Python必須知道的內(nèi)置函數(shù)reduce()、split()、map()等,更多后置函數(shù)請看下文2021-09-09
學(xué)會(huì)這個(gè)炫酷圖表利器pyecharts,還怕不被公司重用?
前段時(shí)間,公司高層要看上半年度項(xiàng)目組業(yè)績數(shù)據(jù)分析,沒辦法,硬著頭皮也要上!說到數(shù)據(jù)分析,肯定離不開數(shù)據(jù)的可視化,畢竟圖表比冷冰冰的數(shù)字更加直觀,Boss只想一眼就能看出趨勢和結(jié)論.今天我們就聊一聊 pyecharts 中幾種常用的圖表, ,需要的朋友可以參考下2021-06-06
利用Python自制網(wǎng)頁并實(shí)現(xiàn)一鍵自動(dòng)生成探索性數(shù)據(jù)分析報(bào)告
這篇文章主要介紹了利用Python自制了網(wǎng)頁并實(shí)現(xiàn)一鍵自動(dòng)生成探索性數(shù)據(jù)分析報(bào)告,文章內(nèi)容具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-05-05
python 實(shí)現(xiàn)數(shù)字字符串左側(cè)補(bǔ)零的方法
今天小編就為大家分享一篇python 實(shí)現(xiàn)數(shù)字字符串左側(cè)補(bǔ)零的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-12-12

