首頁 後端開發 Python教學 Python: 你不知道的 super

Python: 你不知道的 super

Nov 15, 2016 pm 03:14 PM
python

super() 的入門使用

在類別的繼承中,如果重定義某個方法,該方法會覆寫父類別的同名方法,但有時,我們希望能同時實作父類別的功能,這時,我們就需要呼叫父類的方法了,可透過使用 super 來實現,例如:

class Animal(object):
    def __init__(self, name):
        self.name = name
    def greet(self):
        print 'Hello, I am %s.' % self.name

class Dog(Animal):
    def greet(self):
        super(Dog, self).greet()   # Python3 可使用 super().greet()
        print 'WangWang...'
登入後複製

在上面,Animal 是父類,Dog 是子類,我們在Dog 類別重定義了 greet 方法,為了能同時實現父類的功能,我們又呼叫了父類的方法,看下面的使用:

>>> dog = Dog('dog')
>>> dog.greet()
Hello, I am dog.
WangWang..
登入後複製

super 的一個最常見用法可以說是在子類中調用父類的初始化方法了,比如:

class Base(object):
    def __init__(self, a, b):
        self.a = a
        self.b = b

class A(Base):
    def __init__(self, a, b, c):
        super(A, self).__init__(a, b)  # Python3 可使用 super().__init__(a, b)
        self.c = c
登入後複製

深入super()

看了上面的使用,你可能會覺得 super 的使用很簡單,無非就是取得了父類,並呼叫父類的方法。其實,在上面的情況下,super 獲得的類別剛好是父類,但在其他情況就不一定了,super 其實和父類沒有實質的關聯。

讓我們來看一個稍微複雜的例子,涉及到多重繼承,代碼如下:

class Base(object):
    def __init__(self):
        print "enter Base"
        print "leave Base"

class A(Base):
    def __init__(self):
        print "enter A"
        super(A, self).__init__()
        print "leave A"

class B(Base):
    def __init__(self):
        print "enter B"
        super(B, self).__init__()
        print "leave B"

class C(A, B):
    def __init__(self):
        print "enter C"
        super(C, self).__init__()
        print "leave C"
登入後複製

其中,Base 是父類,A, B 繼承自Base, C 繼承自A, B,它們的繼承關係如下:

      Base
      /  \
     /    \
    A      B
     \    /
      \  /
       C
登入後複製

現在,讓我們來看看使用:

>>> c = C()
enter C
enter A
enter B
enter Base
leave Base
leave B
leave A
leave C
登入後複製

如果你認為 super 代表『呼叫父類的方法』,那你很可能會疑惑為什麼enter A 的下一句不是enter Base 而是enter B。原因是,super 和父類沒有實質的關聯,現在讓我們搞清楚 super 是怎麼運作的。

MRO 列表

事實上,對於你定義的每一個類,Python 會計算出一個方法解析順序(Method Resolution Order, MRO)列表,它代表了類繼承的順序,我們可以使用下面的方式獲得某個類別的MRO 列表:

>>> C.mro()   # or C.__mro__ or C().__class__.mro()
[__main__.C, __main__.A, __main__.B, __main__.Base, object]
登入後複製

那這個MRO 列表的順序是怎麼定的呢,它是透過一個 C3 線性化演算法來實現的,這裡我們就不去深究這個演算法了,有興趣的讀者可以自己去了解一下,總的來說,一個類別的MRO 列表就是合併所有父類的MRO 列表,並遵循以下三個原則:

子類永遠在父類前面

如果有多個父類,會根據它們在列表中的順序被檢查

如果對下一個類別有兩個合法的選擇,選擇第一個父類

super 原理

super 的工作原理如下:

def super(cls, inst):
    mro = inst.__class__.mro()
    return mro[mro.index(cls) + 1]
登入後複製

其中,cls 代表類,inst 代表實例,上面的程式碼做了兩件事:

獲取inst 的MRO 列表

查找cls 在當前MRO 列表中的index, 並返回它的下一個類,即mro[index + 1]

當你使用 super( cls, inst) 時,Python 會在inst 的MRO 清單上搜尋cls 的下一個類別。

現在,讓我們回到前面的例子。

首先看類別C 的 __init__ 方法:

super(C, self).__init__()
登入後複製

這裡的self 是目前C 的實例,self.__class__.mro() 結果是:

[__main__.C, __main__.A, __main__.B, __main__.Base, object]
登入後複製

可以看到,C 的下一個類別是A,於是,跳跳可以看到到了A 的 __init__,這時會印出enter A,並執行下面一行程式碼:

super(A, self).__init__()
登入後複製

注意,這裡的self 也是當前C 的實例,MRO 列表跟上面是一樣的,搜尋A 在MRO 中的下一個類,發現是B,於是,跳到了B 的 __init__,這時會印出enter B,而不是enter Base。

整個過程還是比較清晰的,關鍵是要理解 super 的工作方式,而不是想當然地認為 super 呼叫了父類的方法。

小結

事實上,super 和父類沒有實質的關聯。

super(cls, inst) 獲得的是 cls 在 inst 的 MRO 列表中的下一個類別。


本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1664
14
CakePHP 教程
1423
52
Laravel 教程
1318
25
PHP教程
1269
29
C# 教程
1248
24
PHP和Python:解釋了不同的範例 PHP和Python:解釋了不同的範例 Apr 18, 2025 am 12:26 AM

PHP主要是過程式編程,但也支持面向對象編程(OOP);Python支持多種範式,包括OOP、函數式和過程式編程。 PHP適合web開發,Python適用於多種應用,如數據分析和機器學習。

在PHP和Python之間進行選擇:指南 在PHP和Python之間進行選擇:指南 Apr 18, 2025 am 12:24 AM

PHP適合網頁開發和快速原型開發,Python適用於數據科學和機器學習。 1.PHP用於動態網頁開發,語法簡單,適合快速開發。 2.Python語法簡潔,適用於多領域,庫生態系統強大。

sublime怎麼運行代碼python sublime怎麼運行代碼python Apr 16, 2025 am 08:48 AM

在 Sublime Text 中運行 Python 代碼,需先安裝 Python 插件,再創建 .py 文件並編寫代碼,最後按 Ctrl B 運行代碼,輸出會在控制台中顯示。

PHP和Python:深入了解他們的歷史 PHP和Python:深入了解他們的歷史 Apr 18, 2025 am 12:25 AM

PHP起源於1994年,由RasmusLerdorf開發,最初用於跟踪網站訪問者,逐漸演變為服務器端腳本語言,廣泛應用於網頁開發。 Python由GuidovanRossum於1980年代末開發,1991年首次發布,強調代碼可讀性和簡潔性,適用於科學計算、數據分析等領域。

Python vs. JavaScript:學習曲線和易用性 Python vs. JavaScript:學習曲線和易用性 Apr 16, 2025 am 12:12 AM

Python更適合初學者,學習曲線平緩,語法簡潔;JavaScript適合前端開發,學習曲線較陡,語法靈活。 1.Python語法直觀,適用於數據科學和後端開發。 2.JavaScript靈活,廣泛用於前端和服務器端編程。

Golang vs. Python:性能和可伸縮性 Golang vs. Python:性能和可伸縮性 Apr 19, 2025 am 12:18 AM

Golang在性能和可擴展性方面優於Python。 1)Golang的編譯型特性和高效並發模型使其在高並發場景下表現出色。 2)Python作為解釋型語言,執行速度較慢,但通過工具如Cython可優化性能。

vscode在哪寫代碼 vscode在哪寫代碼 Apr 15, 2025 pm 09:54 PM

在 Visual Studio Code(VSCode)中編寫代碼簡單易行,只需安裝 VSCode、創建項目、選擇語言、創建文件、編寫代碼、保存並運行即可。 VSCode 的優點包括跨平台、免費開源、強大功能、擴展豐富,以及輕量快速。

notepad 怎麼運行python notepad 怎麼運行python Apr 16, 2025 pm 07:33 PM

在 Notepad 中運行 Python 代碼需要安裝 Python 可執行文件和 NppExec 插件。安裝 Python 並為其添加 PATH 後,在 NppExec 插件中配置命令為“python”、參數為“{CURRENT_DIRECTORY}{FILE_NAME}”,即可在 Notepad 中通過快捷鍵“F6”運行 Python 代碼。

See all articles