python中類的初始化方法是__init__(),因此父類子類的初始化方法都是這個,如果子類不實現(xiàn)這個函數(shù),初始化時調(diào)用父類的初始化函數(shù),如果子類實現(xiàn)這個函數(shù),就覆蓋了父類的這個函數(shù),既然繼承父類,就要在這個函數(shù)里顯式調(diào)用一下父類的__init__(),這跟C++,jAVA不一樣,他們是自動調(diào)用父類初始化函數(shù)的。
專注于為中小企業(yè)提供成都網(wǎng)站設(shè)計、網(wǎng)站制作服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)大武口免費(fèi)做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動了1000多家企業(yè)的穩(wěn)健成長,幫助中小企業(yè)通過網(wǎng)站建設(shè)實現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。
調(diào)用父類函數(shù)有以下方法:class A:def method(self, arg):pass
class B(A):
def method(self, arg):
# A.method(self,arg) # 1
# super(B, self).method(arg) #2
super().method(arg) #3
1.直接寫類名調(diào)用
2.用super(type,obj).method(arg)方法調(diào)用。
3.在類定義中調(diào)用本類的父類方法,可以直接
super().method(arg).
在代碼中調(diào)用對象的父類的方法的示例:ob = B()super(B,ob).method(arg) #調(diào)用class B的父類class A的method。
初始化中調(diào)用父類初始化方法示例:
class B(A):
1、函數(shù)定義
①使用def關(guān)鍵字定義函數(shù)
②
def 函數(shù)名(參數(shù)1.參數(shù)2.參數(shù)3...):
"""文檔字符串,docstring,用來說明函數(shù)的作用"""
#函數(shù)體
return 表達(dá)式
注釋的作用:說明函數(shù)是做什么的,函數(shù)有什么功能。
③遇到冒號要縮進(jìn),冒號后面所有的縮進(jìn)的代碼塊構(gòu)成了函數(shù)體,描述了函數(shù)是做什么的,即函數(shù)的功能是什么。Python函數(shù)的本質(zhì)與數(shù)學(xué)中的函數(shù)的本質(zhì)是一致的。
2、函數(shù)調(diào)用
①函數(shù)必須先定義,才能調(diào)用,否則會報錯。
②無參數(shù)時函數(shù)的調(diào)用:函數(shù)名(),有參數(shù)時函數(shù)的調(diào)用:函數(shù)名(參數(shù)1.參數(shù)2.……)
③不要在定義函數(shù)的時候在函數(shù)體里面調(diào)用本身,否則會出不來,陷入循環(huán)調(diào)用。
④函數(shù)需要調(diào)用函數(shù)體才會被執(zhí)行,單純的只是定義函數(shù)是不會被執(zhí)行的。
⑤Debug工具中Step into進(jìn)入到調(diào)用的函數(shù)里,Step Into My Code進(jìn)入到調(diào)用的模塊里函數(shù)。
人工智能火了,Python跟著再一次火了。那Python之父了解一下。
吉多·范羅蘇姆1956年1月31日出生于荷蘭。
1982年獲得阿姆斯特丹大學(xué)的數(shù)學(xué)和計算機(jī)科學(xué)的碩士學(xué)位。
1991年初,吉多·范羅蘇姆發(fā)布了Python的第一個公開發(fā)行版。
1995移居到美國。
2005年加入Google公司,其中有一半時間是花在Python上。
2006年,他被美國計算機(jī)協(xié)會(ACM)認(rèn)定為著名工程師。
大師加入谷歌的那一年,他49歲,谷歌老板,拉里佩奇33歲。
子類調(diào)用父類函數(shù)有以下方法:
直接寫類名調(diào)用
用 super(type, obj).method(arg)方法調(diào)用。
在類定義中調(diào)用本類的父類方法,可以直接用super().method(arg)
1
2
3
4
5
6
7
8
9
class A:
def method(self, arg):
pass
class B(A):
def method(self, arg):
# A.method(self,arg) # 1
# super(B, self).method(arg) # 2
super().method(arg) # 3
在繼承關(guān)系中,我們想調(diào)用已經(jīng)被覆蓋了的父類的方法,就需要如下實現(xiàn):
解決方法:
要調(diào)用父類中的方法,就要使用超類(超集)方法super(),該方法旨在調(diào)用已經(jīng)被覆蓋的父類的成員方法。
討論:
有關(guān)python是如何實現(xiàn)繼承的?
針對每一個定義的類,都會計算出一個成為方法解析順序(MRO)的元組,其只是簡單的對所有基類進(jìn)行簡單地線性排列。
通過上述的C類調(diào)用MRO表,我們不難看出,它將本類開始一直到object類直接所有的父類一次性從左向右逐層向上的排列了出來(先排列自己,在排列自己的父類,最后排列父類的父類,以及最后的object)
然而MRO為何如此排列,這里要涉及到一個非常令人討厭的數(shù)學(xué)算法,C3線性化處理,這里只是總結(jié)其三個約束:(簡單點說,其實就是對父類進(jìn)行歸并排列)
1、先檢查子類,再檢查父類
2、有多個父類時,按照MRO表的順序依次查看
3、如果下一個待選的類出現(xiàn)了兩個合法的選擇,那么就從第一個父類中選取。
4、補(bǔ)充一點:MRO對類的排序幾乎適用于任何定義的類層次結(jié)構(gòu)。
來了來了,它真的來了:重點~~
有很多同學(xué)是否仔細(xì)看過上邊的代碼?
有關(guān)super()函數(shù),以下重點需要各位明白:
在重寫的方法中僅使用一次super()方法時,會按照MRO表從下一個類開始搜索對應(yīng)的方法或?qū)傩?,以此類推?所以C中重寫了父類的構(gòu)造,構(gòu)造中有super,所以會按照順序去查找MRO中下一個類的方法,發(fā)現(xiàn)A中也有super,就會再去B中找對應(yīng)的方法(同名方法是__init__),所以找到B的構(gòu)造,可是B中又有super,就會再去MRO中B的下一個類(Base)中找對應(yīng)的方法(Base的__init__()方法),所以會先打印“Base.__init__”,打印完后又因為B的__init__中還有打印“B.__init__”,所以接著打印‘B.__init__’,又因為打印完后A中還有打印“A.__init__”,所以再打印“A.__init__”,最后打印“C.__init__”。這樣就可以遍歷MRO整張表中所有的對應(yīng)的__init__()方法,并且讓每個方法只會被調(diào)用一次。
為了更好的記憶:當(dāng)所有重寫的方法中只使用了一次super函數(shù)時,會從最上層的類依次調(diào)用其指定的方法即可以理解為(object-Base-B-A-C)。
所以,輸出結(jié)果為:
甚至于如下情況更為耐人尋味,仔細(xì)品一品:
值的一提的是:AB均沒有顯式的繼承的父類,為何結(jié)果為打印‘AB’呢?這里就要理解MRO的含義了哦!
子類繼承父類時,如果父類有需要初始化的屬性,那么必須在子類中調(diào)用父類的初始化方法,幫助父類進(jìn)行初始化,否則,子類可以不調(diào)用父類的初始化方法
代碼示例
"""
父類Car中沒有屬性需要初始化,所有子類中也不需要調(diào)用父類的初始化方法
"""
class?Car:
def?show_name(self):
print('car?name')
class?EeleCar(Car):
pass
car?=?EeleCar()
car.show_name()