本文實(shí)例講述了Python實(shí)現(xiàn)子類調(diào)用父類的方法。分享給大家供大家參考。具體實(shí)現(xiàn)方法如下:
python和其他面向?qū)ο笳Z(yǔ)言類似,每個(gè)類可以擁有一個(gè)或者多個(gè)父類,它們從父類那里繼承了屬性和方法。如果一個(gè)方法在子類的實(shí)例中被調(diào)用,或者一個(gè)屬性在子類的實(shí)例中被訪問(wèn),但是該方法或?qū)傩栽谧宇愔胁⒉淮嬖冢敲淳蜁?huì)自動(dòng)的去其父類中進(jìn)行查找。
繼承父類后,就能調(diào)用父類方法和訪問(wèn)父類屬性,而要完成整個(gè)集成過(guò)程,子類是需要調(diào)用的構(gòu)造函數(shù)的。
子類不顯式調(diào)用父類的構(gòu)造方法,而父類構(gòu)造函數(shù)初始化了一些屬性,就會(huì)出現(xiàn)問(wèn)題
如果子類和父類都有構(gòu)造函數(shù),子類其實(shí)是重寫了父類的構(gòu)造函數(shù),如果不顯式調(diào)用父類構(gòu)造函數(shù),父類的構(gòu)造函數(shù)就不會(huì)被執(zhí)行,導(dǎo)致子類實(shí)例訪問(wèn)父類初始化方法中初始的變量就會(huì)出現(xiàn)問(wèn)題。
先來(lái)看看如下示例:
??? def __init__(self):
??????? self.namea="aaa"
??? def funca(self):
??????? print "function a : %s"%self.namea
class B(A):
??? def __init__(self):
??????? self.nameb="bbb"
??? def funcb(self):
??????? print "function b : %s"%self.nameb
b=B()
print b.nameb
b.funcb()
b.funca()
運(yùn)行結(jié)果:
function b : bbb
Traceback (most recent call last):
? File "D:workbenchpythonMyPythonProjectteststudyoverwrite_method.py", line 19, in
??? print b.funca()
? File "D:workbenchpythonMyPythonProjectteststudyoverwrite_method.py", line 6, in funca
??? print "function a : %s"%self.namea
AttributeError: B instance has no attribute 'namea'
在子類中,構(gòu)造函數(shù)被重寫,但新的構(gòu)造方法沒(méi)有任何關(guān)于初始化父類的namea屬性的代碼,為了達(dá)到預(yù)期的效果,子類的構(gòu)造方法必須調(diào)用其父類的構(gòu)造方法來(lái)進(jìn)行基本的初始化。有兩種方法能達(dá)到這個(gè)目的:調(diào)用超類構(gòu)造方法的未綁定版本,或者使用super函數(shù)。
方法一:調(diào)用未綁定的超類構(gòu)造方法
修改代碼,多增一行:
??? def __init__(self):
??????? self.namea="aaa"
??? def funca(self):
??????? print "function a : %s"%self.namea
class B(A):
??? def __init__(self):
??????? #這一行解決了問(wèn)題
??????? A.__init__(self)
??????? self.nameb="bbb"
??? def funcb(self):
??????? print "function b : %s"%self.nameb
b=B()
print b.nameb
b.funcb()
b.funca()
如上有注釋的一行解決了該問(wèn)題,直接使用父類名稱調(diào)用其構(gòu)造函數(shù)即可。
這種方法叫做調(diào)用父類的未綁定的構(gòu)造方法。在調(diào)用一個(gè)實(shí)例的方法時(shí),該方法的self參數(shù)會(huì)被自動(dòng)綁定到實(shí)例上(稱為綁定方法)。但如果直接調(diào)用類的方法(比如A.__init),那么就沒(méi)有實(shí)例會(huì)被綁定。這樣就可以自由的提供需要的self參數(shù),這種方法稱為未綁定unbound方法。
通過(guò)將當(dāng)前的實(shí)例作為self參數(shù)提供給未綁定方法,B類就能使用其父類構(gòu)造方法的所有實(shí)現(xiàn),從而namea變量被設(shè)置。
方法二:使用super函數(shù)
修改代碼,這次需要增加在原來(lái)代碼上增加2行:
class A(object):
??? def __init__(self):
??????? self.namea="aaa"
??? def funca(self):
??????? print "function a : %s"%self.namea
class B(A):
??? def __init__(self):
??????? #這一行解決問(wèn)題
??????? 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()
如上有注釋的為新增的代碼,其中第一句讓類A繼承自object類,這樣才能使用super函數(shù),因?yàn)檫@是python的“新式類”支持的特性。當(dāng)前的雷和對(duì)象可以作為super函數(shù)的參數(shù)使用,調(diào)用函數(shù)返回的對(duì)象的任何方法都是調(diào)用超類的方法,而不是當(dāng)前類的方法。
super函數(shù)會(huì)返回一個(gè)super對(duì)象,這個(gè)對(duì)象負(fù)責(zé)進(jìn)行方法解析,解析過(guò)程其會(huì)自動(dòng)查找所有的父類以及父類的父類。
方法一更直觀,方法二可以一次初始化所有超類
super函數(shù)比在超累中直接調(diào)用未綁定方法更直觀,但是其最大的有點(diǎn)是如果子類繼承了多個(gè)父類,它只需要使用一次super函數(shù)就可以。然而如果沒(méi)有這個(gè)需求,直接使用A.__init__(self)更直觀一些。
希望本文所述對(duì)大家的Python程序設(shè)計(jì)有所幫助。
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對(duì)您有幫助就好】元
