python?中的?super詳解
提到 super
,最直接的想法就是它代表了父類(lèi),替父類(lèi)執(zhí)行某些方法。但是理解也僅止步于此,下面對(duì) super
做進(jìn)一步理解
super 的完整形式
常見(jiàn)的 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í)行了父類(lèi)的初始化方法,也就繼承了父類(lèi)的 name 屬性。
但是其實(shí) super
的完整形式為
super(Male, self).__init__('xiaoming')
super
是一個(gè)類(lèi),其中第二個(gè)參數(shù)是個(gè) class 或者 object,決定了使用怎樣的 mro。第一個(gè)參數(shù)是個(gè) class,決定了從 mro 哪個(gè) class 后面的 class 開(kāi)始尋找,并將函數(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
行的語(yǔ)句等價(jià)于
# super(Male, self).__init__('xiaoming') Person.__init__(self,'xiaoming')
執(zhí)行結(jié)果同上
super 的使用
super
可以在定義類(lèi)之外的地方使用
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í)行了父類(lèi)的 __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í)候繞過(guò)了 Person
,只輸出了 Animal
和 Male
。而在類(lèi)之外執(zhí)行的 super
,執(zhí)行了 Male
的父類(lèi)(Person、Animal)的 __init__
方法。 說(shuō)明了 2 點(diǎn):
super
的第一個(gè)參數(shù)決定了選擇self
的 mro 哪個(gè) class 之后的 class。super
可以在類(lèi)定義之外執(zhí)行。
再看一個(gè)例子將會(huì)更加明白
直覺(jué)上來(lái)說(shuō),D
的實(shí)例會(huì)執(zhí)行父類(lèi)的 say()
,首先會(huì)找到 B
,于是會(huì)執(zhí)行 B
的父類(lèi)的 say()
,于是會(huì)輸出 'A'
。結(jié)果卻是 'C'
,原因就是 self
代表了 D
的實(shí)例,而 D
的 mro 是 ['B','C','A']
,D
的實(shí)例執(zhí)行父類(lèi)的 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'
類(lèi)中使用 super
的時(shí)候,可以省略參數(shù)而直接寫(xiě)成 super()
,這時(shí) super 會(huì)將他所在的類(lèi)當(dāng)作第一個(gè)參數(shù),將所在函數(shù)的第一個(gè)參數(shù)當(dāng)作自己的第二個(gè)參數(shù)。顯然,這樣省略參數(shù)的 super
不能在類(lèi)之外直接使用。
最后,查看一個(gè)類(lèi)的 mro 可以用 class.__mro__
或者 class.mro()
獲取
到此這篇關(guān)于python 中的 super的文章就介紹到這了,更多相關(guān)python super內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python史上最全種類(lèi)數(shù)據(jù)庫(kù)操作方法分享
本文將詳細(xì)探討如何在Python中連接全種類(lèi)數(shù)據(jù)庫(kù)以及實(shí)現(xiàn)相應(yīng)的CRUD(創(chuàng)建,讀取,更新,刪除)操作,文中的示例代碼講解詳細(xì),需要的可以參考一下2023-07-07如何在mac環(huán)境中用python處理protobuf
這篇文章主要介紹了如何在mac環(huán)境中用python處理protobuf,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12python獲取酷狗音樂(lè)top500的下載地址 MP3格式
這篇文章主要介紹了python獲取酷狗音樂(lè)top500的下載地址 MP3格式,文中給大家提到了python--爬取酷狗TOP500的數(shù)據(jù),需要的朋友可以參考下2018-04-04python語(yǔ)言time庫(kù)和datetime庫(kù)基本使用詳解
這篇文章主要介紹了python語(yǔ)言time庫(kù)和datetime庫(kù)基本使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12python生成tensorflow輸入輸出的圖像格式的方法
本篇文章主要介紹了python生成tensorflow輸入輸出的圖像格式的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-02-02Python類(lèi)的詳細(xì)定義與使用案例(實(shí)例講解)
這篇文章主要給大家介紹了關(guān)于Python類(lèi)的詳細(xì)定義與使用案例的相關(guān)資料,在Python中類(lèi)表示具有相同屬性和方法的對(duì)象的集合,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-10-10