在對于python中類的使用上,我們分出了子類和父類兩種。對于這二者之間的關(guān)系,我們可以簡單理解為繼承。不過python中加入了實例的討論,那么對于繼承后的子類來說,父類的實例是否被繼承又是我們所需要思考的問題。下面我們就子類和父類進(jìn)行簡單介紹,然后就二者之間的繼承關(guān)系重點分析。
創(chuàng)新互聯(lián)建站公司2013年成立,先為鶴城等服務(wù)建站,鶴城等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢服務(wù)。為鶴城企業(yè)網(wǎng)站制作PC+手機+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問題。
1.概念
子類和父類主要描述的是類之間的繼承關(guān)系,即所屬關(guān)系。繼承的類可在被繼承的類的基礎(chǔ)上添加格外的參數(shù)和行為,新類稱為子類,擴展類;被繼承的類稱為基類、父類或者超類。
2.繼承關(guān)系
子類與父類的關(guān)系是 “is” 的關(guān)系,如上 Cat 繼承于 Animal 類,我們可以說:
“A”是 Animal 類的實例,但,“A”不是 Cat 類的實例。
“C”是 Animal 類的實例,“C”也是 Cat 類的實例。
判斷對象之間的關(guān)系,我們可以通過 isinstance (變量,類型) 來進(jìn)行判斷:
print('"A" IS Animal?', isinstance(A, Animal))
print('"A" IS Cat?', isinstance(A, Cat))
print('"C" IS Animal?', isinstance(C, Animal))
print('"C" IS Cat?', isinstance(C, Cat))
Python實現(xiàn)子類調(diào)用父類的方法
python和其他面向?qū)ο笳Z言類似,每個類可以擁有一個或者多個父類,它們從父類那里繼承了屬性和方法。如果一個方法在子類的實例中被調(diào)用,或者一個屬性在子類的實例中被訪問,但是該方法或?qū)傩栽谧宇愔胁⒉淮嬖?,那么就會自動的去其父類中進(jìn)行查找。
繼承父類后,就能調(diào)用父類方法和訪問父類屬性,而要完成整個集成過程,子類是需要調(diào)用的構(gòu)造函數(shù)的。
子類不顯式調(diào)用父類的構(gòu)造方法,而父類構(gòu)造函數(shù)初始化了一些屬性,就會出現(xiàn)問題
如果子類和父類都有構(gòu)造函數(shù),子類其實是重寫了父類的構(gòu)造函數(shù),如果不顯式調(diào)用父類構(gòu)造函數(shù),父類的構(gòu)造函數(shù)就不會被執(zhí)行,導(dǎo)致子類實例訪問父類初始化方法中初始的變量就會出現(xiàn)問題。
在繼承關(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、補充一點: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的含義了哦!
兩種方法解決:
- 第一種
class A:
def __init__(self):
self.namea="aaa"
def funca(self):
print "function a : %s"%self.namea
class B(A):
def __init__(self):
#這一行解決了問題
A.__init__(self)
self.nameb="bbb"
def funcb(self):
print "function b : %s"%self.nameb
b=B()
print b.nameb
b.funcb()
b.funca()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
第二種:
#父類需要繼承object對象
class A(object):
def __init__(self):
self.namea="aaa"
def funca(self):
print "function a : %s"%self.namea
class B(A):
def __init__(self):
#這一行解決問題
super(B,self).__init__()
self.nameb="bbb"
def funcb(self):
print "function b : %s"%self.nameb
b=B()
print b.nameb
b.funcb()
b.funca()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
PS:讓類A繼承自object類,這樣才能使用super函數(shù),因為這是python的“新式類”支持的特性。當(dāng)前的class和對象可以作為super函數(shù)的參數(shù)使用,調(diào)用函數(shù)返回的對象的任何方法都是調(diào)用超類的方法,而不是當(dāng)前類的方法。
優(yōu)劣:
- 方法一更直觀,方法二可以一次初始化所有超類
- super函數(shù)比在超類中直接調(diào)用未綁定方法更直觀,但是其最大的優(yōu)點是如果子類繼承了多個父類,它只需要使用一次super函數(shù)就可以。然而如果沒有這個需求,直接使用A.init(self)更直觀一些。
子類調(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