亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Python super( )函數(shù)用法總結(jié)

瀏覽:4日期:2022-06-17 08:28:09
目錄一、super( ) 的用途二、了解 super 的基礎(chǔ)信息三、典型用法3.1 單繼承問(wèn)題3.2 單繼承問(wèn)題拓展3.3 重復(fù)調(diào)用問(wèn)題3.4 super(type) 問(wèn)題一、super( ) 的用途

了解 super() 函數(shù)之前,我們首先要知道 super() 的用途是啥?

主要用來(lái)在子類(lèi)中調(diào)用父類(lèi)的方法。 多用于多繼承問(wèn)題中,解決查找順序(MRO)、重復(fù)調(diào)用(鉆石繼承)等種種問(wèn)題。二、了解 super 的基礎(chǔ)信息

語(yǔ)法格式:

super([type[, object-or-type]])

函數(shù)描述:

返回一個(gè)代理對(duì)象,它會(huì)將方法調(diào)用委托給 type 的父類(lèi)或兄弟類(lèi)。

參數(shù)說(shuō)明:

type —— 類(lèi),可選參數(shù)。object-or-type —— 對(duì)象或類(lèi),一般是 self,可選參數(shù)。

返回值:

super object —— 代理對(duì)象。

help 幫助信息:

>>> help(super)Help on class super in module builtins: class super(object) | super() -> same as super(__class__, <first argument>) | super(type) -> unbound super object | super(type, obj) -> bound super object; requires isinstance(obj, type) | super(type, type2) -> bound super object; requires issubclass(type2, type) | Typical use to call a cooperative superclass method: | class C(B): | def meth(self, arg): | super().meth(arg) | This works for class methods too: | class C(B): | @classmethod | def cmeth(cls, arg): | super().cmeth(arg) ... ... super 是一個(gè)繼承自 object 的類(lèi),調(diào)用 super() 函數(shù)其實(shí)就是 super 類(lèi)的實(shí)例化。 根據(jù)官方文檔的解釋 super() 函數(shù)返回的對(duì)象 —— super object,就是一個(gè)代理對(duì)象。 super() 有四種參數(shù)的組合形式。 super()適用于類(lèi)的靜態(tài)方法。三、典型用法3.1 單繼承問(wèn)題

首先我們看一個(gè)最基本的子類(lèi)調(diào)用父類(lèi)方法的示例:

>>> class A:def funxx(self): print('執(zhí)行 A 中的 funxx 方法 ... ...') >>> class B(A):def funxx(self): A.funxx(self) # 通過(guò)類(lèi)名調(diào)用父類(lèi)中的同名方法,self 參數(shù)代表 B 類(lèi)的實(shí)例對(duì)象 b print('執(zhí)行 B 中的 funxx 方法 ... ...') >>> b = B()>>> b.funxx()執(zhí)行 A 中的 funxx 方法 ... ...執(zhí)行 B 中的 funxx 方法 ... ... 定義一個(gè)繼承自 A 類(lèi)的子類(lèi) B,并在 B 類(lèi)中重寫(xiě) funxx() 方法,B 中的 funxx() 是對(duì) A 中的 funxx() 功能的拓展。 因?yàn)槭峭卣沽?A 類(lèi)的 funxx() 方法的功能,所以其任然保留了原功能,即要在子類(lèi) B 中調(diào)用父類(lèi)的同名方法來(lái)實(shí)現(xiàn)原有功能。 上面的示例中是通過(guò) A 類(lèi)類(lèi)名調(diào)用 A 類(lèi)中的同名方法來(lái)實(shí)現(xiàn)的,而第一個(gè)參數(shù) self 實(shí)際傳遞的是 B 類(lèi)的實(shí)例 b。

使用 super() 函數(shù)來(lái)實(shí)現(xiàn)父類(lèi)方法的調(diào)用:

>>> class A:def funxx(self): print('執(zhí)行 A 中的 funxx 方法 ... ...') >>> class B(A):def funxx(self): super().funxx() print('執(zhí)行 B 中的 funxx 方法 ... ...') >>> b = B()>>> b.funxx()執(zhí)行 A 中的 funxx 方法 ... ...執(zhí)行 B 中的 funxx 方法 ... ... 通過(guò)執(zhí)行的結(jié)果可以看出實(shí)現(xiàn)了和普通類(lèi)名調(diào)用的結(jié)果是一樣的。 在具有單繼承的類(lèi)層級(jí)結(jié)構(gòu)中,super 引用父類(lèi)而不必顯式地指定它們的名稱(chēng),從而令代碼更易維護(hù)。(官方文檔描述) 也就是說(shuō),在子類(lèi)中不再用父類(lèi)名調(diào)用父類(lèi)方法,而是用一個(gè)代理對(duì)象調(diào)用父類(lèi)方法,這樣當(dāng)父類(lèi)名改變或者繼承關(guān)系發(fā)生變化時(shí),不用對(duì)每個(gè)調(diào)用處都進(jìn)行修改。3.2 單繼承問(wèn)題拓展

在 help() 的幫助信息中,也說(shuō)明了類(lèi)中使用 super() 不帶參數(shù)的形式等同于 super(__class__, <first argument>) 這種形式。這也是 Python 2.x 和 Python 3.x 關(guān)于 super() 的區(qū)別。

改寫(xiě)之前的單繼承問(wèn)題的代碼:

>>> class A:def funxx(self): print('執(zhí)行 A 中的 funxx 方法 ... ...') >>> class B(A):def funxx(self):super(B, self).funxx()print('執(zhí)行 B 中的 funxx 方法 ... ...') >>> b = B()>>> b.funxx()執(zhí)行 A 中的 funxx 方法 ... ...執(zhí)行 B 中的 funxx 方法 ... ... 基本的調(diào)用方法 A.funxx(self) ,其中 self 指代實(shí)例對(duì)象 b。用語(yǔ)言描述為:實(shí)例對(duì)象 b 通過(guò) A 類(lèi)名調(diào)用方法 funxx()。 官方描述:返回一個(gè)代理對(duì)象,它會(huì)將方法調(diào)用委托給 type 的父類(lèi)或兄弟類(lèi)。用語(yǔ)言描述為:代理對(duì)象 super 通過(guò) type 的父類(lèi)或兄弟類(lèi)調(diào)用其中的方法。 我們發(fā)現(xiàn) super 是通過(guò)參數(shù)設(shè)置來(lái)選擇調(diào)用哪個(gè)父類(lèi)的方法。其中第二個(gè)參數(shù)給出 MRO(方法解析順序),也就是搜索目標(biāo)方法的順序,第一個(gè)參數(shù)則給出搜索目標(biāo)方法的范圍。 例如 super(B, self) ,第一個(gè)參數(shù)為 B,第二個(gè)參數(shù) self 為實(shí)例 b,其所在類(lèi)的繼承順序(MRO)為:B→A→object。所以調(diào)用時(shí)是在 B 的父類(lèi) A 中尋找,如找不到目標(biāo)方法則會(huì)在更上一層的 object 中尋找。

示例:

class A: pass class B(A): pass class C(A): def funxx(self):print('找到 funxx() 位于 C 中...') class D(A): pass class E(B, C): pass class F(E, D): def funff(self):print('執(zhí)行 F 中的 funff()...')super(E, self).funxx() print(f'F 類(lèi)的 MRO : {F.__mro__}')f = F()f.funff()

運(yùn)行結(jié)果:

F 類(lèi)的 MRO : (<class ’__main__.F’>, <class ’__main__.E’>, <class ’__main__.B’>, <class ’__main__.C’>, <class ’__main__.D’>, <class ’__main__.A’>, <class ’object’>)執(zhí)行 F 中的 funff()...找到 funxx() 位于 C 中... 我們可以看出 F 類(lèi)的 MRO:F→E→B→C→D→A→object。 super() 函數(shù)的第一個(gè)參數(shù)為:E,目標(biāo)是調(diào)用 E 類(lèi)的父類(lèi) B 中的 funxx() 方法,可惜 B 類(lèi)中沒(méi)找到,在 B 類(lèi)的兄弟類(lèi) C 中找到了,符合要求。3.3 重復(fù)調(diào)用問(wèn)題

重復(fù)調(diào)用問(wèn)題 也稱(chēng) 鉆石繼承問(wèn)題 或 菱形圖問(wèn)題。

先來(lái)看看普通調(diào)用方法在:

>>> class A:def __init__(self): print('打印屬性 a') >>> class B(A):def __init__(self): print('打印屬性 b') A.__init__(self) >>> class C(A):def __init__(self): print('打印屬性 c') A.__init__(self) >>> class D(B, C):def __init__(self): print('打印屬性 d') B.__init__(self) C.__init__(self) >>> d = D()打印屬性 d打印屬性 b打印屬性 a打印屬性 c打印屬性 a 因?yàn)?B,C 都繼承自 A,所以當(dāng) D 在實(shí)例化時(shí),A 的構(gòu)造函數(shù)被執(zhí)行了兩次。這就是所謂的重復(fù)調(diào)用問(wèn)題。 很顯然,我們只需要調(diào)用一次就可以了,重復(fù)的調(diào)用只會(huì)造成資源浪費(fèi)。

接下來(lái)我們使用 super() 函數(shù)來(lái)調(diào)用:

>>> class A:def __init__(self): print('打印屬性 a') >>> class B(A):def __init__(self): print('打印屬性 b') super().__init__()# super() 等同于 super(B, self) >>> class C(A):def __init__(self): print('打印屬性 c') super().__init__()# super() 等同于 super(C, self) >>> class D(B, C):def __init__(self): print('打印屬性 d') super(D, self).__init__() >>> d = D()打印屬性 d打印屬性 b打印屬性 c打印屬性 a 查看輸出結(jié)果我們發(fā)現(xiàn)雖然解決了重復(fù)調(diào)用問(wèn)題,但是輸出結(jié)果的順序好像與我們想的有所區(qū)別。我們的慣性思維是:先執(zhí)行 D 類(lèi)的 __init__() 方法,接著調(diào)用 B 類(lèi)的 __init__() 方法,B 類(lèi)的構(gòu)造方法中又調(diào)用了父類(lèi) A 的 __init_() 方法,然后再是調(diào)用 C 類(lèi)的 __init_() 方法,該方法也調(diào)用了父類(lèi) A 的 __init__() 方法。所以執(zhí)行的結(jié)果應(yīng)該是:打印屬性 d,打印屬性 b,打印屬性 a,打印屬性 c。 為何結(jié)果不是我們想的那樣呢,首先我們要知道 D 類(lèi)中的第二個(gè)參數(shù) self 為 D 的實(shí)例 d,它提供的 MRO 為:D→B→C→A→object。所以 D 類(lèi)中的 super() 函數(shù)產(chǎn)生的是 d 的代理對(duì)象,當(dāng)其調(diào)用父類(lèi) B 的 __init__() 時(shí),B 的 super() 的第二個(gè)參數(shù)為 D 中的 super object,其所提供的 MRO 依舊為:D→B→C→A→object。也就是說(shuō) B 中的 super() 調(diào)用的是它的上一級(jí) C 中的 __init__() ,而不是 A 中的 __init__()。所以執(zhí)行的結(jié)果是:打印屬性 d,打印屬性 b,打印屬性 c,打印屬性 a。3.4 super(type) 問(wèn)題

>>> class A: def funxx(self): print('...A...') >>> class B(A): def funxx(self): print('...B...') >>> sa = super(B)>>> print(sa)<super: <class ’B’>, NULL>>>> print(type(sa))<class ’super’>

可以看出 super(type) 返回的是一個(gè)無(wú)效的對(duì)象,或者是未綁定的 super object。

到此這篇關(guān)于Python super( )函數(shù)用法總結(jié)的文章就介紹到這了,更多相關(guān)super( )函數(shù)內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Python 編程
相關(guān)文章:
主站蜘蛛池模板: 国产美女网站视频 | 国产精品污 | 成年免费大片黄在线观看一 | 午夜男男xx00视频免费 | 国产三级91 | 亚洲综合偷自成人网第页 | 九九精品免费观看在线 | 中文国产成人精品久久无广告 | 日本免费黄视频 | 国产精品自线在线播放 | 久久久久久亚洲精品影院 | 日韩免费一级毛片欧美一级日韩片 | 91久久国产综合精品女同国语 | 亚洲精品国产福利在线观看 | 亚洲国产成人精品91久久久 | 成人精品免费视频 | 国产妞干网 | 中国一级特黄毛片 | 成人做爰毛片免费视频 | 国产免费一级精品视频 | 日本免费人成黄页网观看视频 | 肉色网站 | 久久一区精品 | 成人合成mv福利视频网站 | 一级毛片免费的 | 亚洲精品美女国产一区 | 国产福利在线永久视频 | 69久成人做爰视频 | 国产精品高清一区二区三区 | 韩国日本美国免费毛片 | 亚洲毛片在线观看 | 在线视频观看免费视频18 | 97超级碰久久久久香蕉人人 | 久久er这里都是精品23 | 91仓本c仔约高挑嫩模在线 | 免费影院在线 | 久久精品国产国产精品四凭 | 中国一级一级全黄 | 二区在线观看 | 高清影院|精品秒播3 | 成人毛片网 |