網站首頁 編程語言 正文
提到 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 是 Male
,Person
,Animal
,Object
。Male
在實例化的時候執行了父類的 __init__
方法,而此時 super
的第一個參數是 Person
,于是使用 Person
后面的 Animal
,而 Animal
的 __init__
方法只有一個參數,super
卻傳遞了2個參數,于是報錯了。正確地修改為
# class Person: super(Person,self).__init__(name)
執行結果為
可以看到 Male
實例化的時候繞過了 Person
,只輸出了 Animal
和 Male
。而在類之外執行的 super
,執行了 Male
的父類(Person、Animal)的 __init__
方法。 說明了 2 點:
-
super
的第一個參數決定了選擇self
的 mro 哪個 class 之后的 class。 -
super
可以在類定義之外執行。
再看一個例子將會更加明白
直覺上來說,D
的實例會執行父類的 say()
,首先會找到 B
,于是會執行 B
的父類的 say()
,于是會輸出 'A'
。結果卻是 'C'
,原因就是 self
代表了 D
的實例,而 D
的 mro 是 ['B','C','A']
,D
的實例執行父類的 say()
,會找到 B
執行 B
的 super
方法,相當于 super(B,self).say()
,而此時的 self
代表 D
,mro 搜索會選擇 B
后面的 class 也就是 C
,執行 C
的 say()
,于是最終結果輸出 'C'
類中使用 super
的時候,可以省略參數而直接寫成 super()
,這時 super 會將他所在的類當作第一個參數,將所在函數的第一個參數當作自己的第二個參數。顯然,這樣省略參數的 super
不能在類之外直接使用。
最后,查看一個類的 mro 可以用 class.__mro__
或者 class.mro()
獲取
原文鏈接:https://blog.csdn.net/qq_26826585/article/details/126480616
相關推薦
- 2022-06-24 Windows?Server?2012遠程默認端口3389的修改方法_win服務器
- 2022-05-04 Spring.Net框架簡介_實用技巧
- 2022-04-19 Android實現一個倒計時自定義控件_Android
- 2022-12-24 Python中切片操作的示例詳解_python
- 2022-07-03 c語言中的二級指針做函數參數說明_C 語言
- 2022-08-11 GoFrame框架數據校驗之校驗結果Error接口對象_Golang
- 2022-11-14 Asp.net?Core項目配置HTTPS支持_實用技巧
- 2024-01-09 idea如何設置自動換行
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支