日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學無先后,達者為師

網站首頁 編程語言 正文

python?中的?super詳解_python

作者:你不是葉秋 ? 更新時間: 2022-10-22 編程語言

提到 super,最直接的想法就是它代表了父類,替父類執行某些方法。但是理解也僅止步于此,下面對 super 做進一步理解

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__)

以上執行結果為

在這里插入圖片描述

這個結果也符合理解,Male 繼承了 Person,在初始化的時候執行了父類的初始化方法,也就繼承了父類的 name 屬性。

但是其實 super 的完整形式為

super(Male, self).__init__('xiaoming')

super 是一個,其中第二個參數是個 class 或者 object,決定了使用怎樣的 mro。第一個參數是個 class,決定了從 mro 哪個 class 后面的 class 開始尋找,并將函數綁定到第二個參數上。兩個參數都是可選的。

本例中,self 就是 Male 的實例對象,于是 self 的 mro 就是 [Male,Person,Object],而第一個參數是 Male,于是就使用 Male 后面的 Person,發現 Person__init__ 函數,于是就只執行 Person__init__ 函數,也就是 super 行的語句等價于

# super(Male, self).__init__('xiaoming')
Person.__init__(self,'xiaoming')

執行結果同上

在這里插入圖片描述

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__)

執行結果為

在這里插入圖片描述

可以看到 16 行報錯了,報錯的原因就是此時的 self 代表的是 Male 實例,Male 的 mro 是 MalePersonAnimalObjectMale 在實例化的時候執行了父類的 __init__ 方法,而此時 super 的第一個參數是 Person,于是使用 Person 后面的 Animal,而 Animal__init__ 方法只有一個參數,super 卻傳遞了2個參數,于是報錯了。正確地修改為

# class Person:
super(Person,self).__init__(name)

執行結果為

在這里插入圖片描述

可以看到 Male 實例化的時候繞過了 Person,只輸出了 AnimalMale。而在類之外執行的 super,執行了 Male 的父類(Person、Animal)的 __init__ 方法。 說明了 2 點:

  1. super 的第一個參數決定了選擇 self 的 mro 哪個 class 之后的 class。
  2. super 可以在類定義之外執行。

再看一個例子將會更加明白

在這里插入圖片描述

直覺上來說,D 的實例會執行父類的 say() ,首先會找到 B,于是會執行 B 的父類的 say(),于是會輸出 'A'。結果卻是 'C',原因就是 self 代表了 D 的實例,而 D 的 mro 是 ['B','C','A']D 的實例執行父類的 say() ,會找到 B 執行 Bsuper 方法,相當于 super(B,self).say(),而此時的 self 代表 D,mro 搜索會選擇 B 后面的 class 也就是 C,執行 Csay(),于是最終結果輸出 'C'

類中使用 super 的時候,可以省略參數而直接寫成 super()這時 super 會將他所在的類當作第一個參數,將所在函數的第一個參數當作自己的第二個參數。顯然,這樣省略參數的 super 不能在類之外直接使用。

最后,查看一個類的 mro 可以用 class.__mro__ 或者 class.mro() 獲取

在這里插入圖片描述

原文鏈接:https://blog.csdn.net/qq_26826585/article/details/126480616

欄目分類
最近更新