本文實例講述了Python實現子類調用父類的方法。分享給大家供大家參考。具體實現方法如下:
python和其他面向對象語言類似,每個類可以擁有一個或者多個父類,它們從父類那里繼承了屬性和方法。如果一個方法在子類的實例中被調用,或者一個屬性在子類的實例中被訪問,但是該方法或屬性在子類中并不存在,那么就會自動的去其父類中進行查找。
繼承父類后,就能調用父類方法和訪問父類屬性,而要完成整個集成過程,子類是需要調用的構造函數的。
子類不顯式調用父類的構造方法,而父類構造函數初始化了一些屬性,就會出現問題
如果子類和父類都有構造函數,子類其實是重寫了父類的構造函數,如果不顯式調用父類構造函數,父類的構造函數就不會被執行,導致子類實例訪問父類初始化方法中初始的變量就會出現問題。
先來看看如下示例:
??? 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()
運行結果:
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'
在子類中,構造函數被重寫,但新的構造方法沒有任何關于初始化父類的namea屬性的代碼,為了達到預期的效果,子類的構造方法必須調用其父類的構造方法來進行基本的初始化。有兩種方法能達到這個目的:調用超類構造方法的未綁定版本,或者使用super函數。
方法一:調用未綁定的超類構造方法
修改代碼,多增一行:
??? 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()
如上有注釋的一行解決了該問題,直接使用父類名稱調用其構造函數即可。
這種方法叫做調用父類的未綁定的構造方法。在調用一個實例的方法時,該方法的self參數會被自動綁定到實例上(稱為綁定方法)。但如果直接調用類的方法(比如A.__init),那么就沒有實例會被綁定。這樣就可以自由的提供需要的self參數,這種方法稱為未綁定unbound方法。
通過將當前的實例作為self參數提供給未綁定方法,B類就能使用其父類構造方法的所有實現,從而namea變量被設置。
方法二:使用super函數
修改代碼,這次需要增加在原來代碼上增加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):
??????? #這一行解決問題
??????? 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函數,因為這是python的“新式類”支持的特性。當前的雷和對象可以作為super函數的參數使用,調用函數返回的對象的任何方法都是調用超類的方法,而不是當前類的方法。
super函數會返回一個super對象,這個對象負責進行方法解析,解析過程其會自動查找所有的父類以及父類的父類。
方法一更直觀,方法二可以一次初始化所有超類
super函數比在超累中直接調用未綁定方法更直觀,但是其最大的有點是如果子類繼承了多個父類,它只需要使用一次super函數就可以。然而如果沒有這個需求,直接使用A.__init__(self)更直觀一些。
希望本文所述對大家的Python程序設計有所幫助。
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061
微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元

