淺析python繼承與多重繼承
記住以下幾點:
直接子類化內(nèi)置類型(如dict,list或str)容易出錯,因為內(nèi)置類型的方法通常會忽略用戶覆蓋的方法,不要子類化內(nèi)置類型,用戶自定義的類應該繼承collections模塊。
def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) # 錯誤案例 class AnswerDict(dict): def __getitem__(self, item): # 錯誤案例 return 42 import collections class DoppelDict2(collections.UserDict): # 正確案例 def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) class AnswerDict2(collections.UserDict): # 正確案例 def __getitem__(self, item): return 42
多重繼承有關(guān)的另一個問題就是:如果同級別的超類定義了同名屬性.Python如何確定使用哪個?
class DoppelDict(dict): def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) class AnswerDict(dict): def __getitem__(self, item): return 42 import collections class DoppelDict2(collections.UserDict): def __setitem__(self, key, value): super().__setitem__(key, [value] * 2) class AnswerDict2(collections.UserDict): def __getitem__(self, item): return 42 class A: def ping(self): print('Ping:', self) class B(A): def pong(self): print('pong:', self) class C(A): def pong(self): print('PONG:', self) class D(B, C): def ping(self): super().ping() print('post-ping:', self) def pingpong(self): self.ping() super().ping() self.pong() super().pong() C.pong(self) if __name__ == '__main__': d = D() print(d.pong()) # 輸出來源于B print(C.pong(d)) #輸出來源于C 超類的方法都可以直接調(diào)用,此時要把實例作為顯示參數(shù)傳入.
python能區(qū)別調(diào)用的是哪個方法,通過方法解析順序
>>> D.mro()
[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
若想把方法調(diào)用委托給超類,推薦的方式是使用內(nèi)置的super()函數(shù).
以下是對于d.pingpong()方法的解讀
>>> self.ping()
Ping: <__main__.D object at 0x000002213877F2B0> post-ping: <__main__.D object at 0x000002213877F2B0> 第一個調(diào)用的是self.ping(),運行的是是D類的ping,方法.
第二個調(diào)用的的是super().ping(),跳過D類的ping方法,找到A類的ping方法.Ping: <__main__.D object at 0x000002213877F2B0>
第三個調(diào)用的是self.pong()方法,根據(jù)__mro__,找到B類實現(xiàn)的pong方法. pong: <__main__.D object at 0x000002213877F2B0>
第四個調(diào)用時super().pong(),也是根據(jù)__mro__,找到B類實現(xiàn)的pong方法. pong: <__main__.D object at 0x000002213877F2B0>
第五個調(diào)用的是C.pong(self),忽略了__mro__,找到的是C類實現(xiàn)的pong方法. PONG: <__main__.D object at 0x000002213877F2B0>
相關(guān)文章
python基于celery實現(xiàn)異步任務周期任務定時任務
這篇文章主要介紹了python基于celery實現(xiàn)異步任務周期任務定時任務,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-12-12如何在keras中添加自己的優(yōu)化器(如adam等)
這篇文章主要介紹了在keras中實現(xiàn)添加自己的優(yōu)化器(如adam等)方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-06-06typing.Dict和Dict的區(qū)別及它們在Python中的用途小結(jié)
當在 Python 函數(shù)中聲明一個 dictionary 作為參數(shù)時,我們一般會把 key 和 value 的數(shù)據(jù)類型聲明為全局變量,而不是局部變量。,這篇文章主要介紹了typing.Dict和Dict的區(qū)別及它們在Python中的用途小結(jié),需要的朋友可以參考下2023-06-06Python使用Marshmallow輕松實現(xiàn)序列化和反序列化
這篇文章主要為大家詳細介紹了Python如何使用Marshmallow輕松實現(xiàn)序列化和反序列化,文中的示例代碼講解詳細,感興趣的小伙伴可以了解下2025-03-03聊聊prod()與cumprod()區(qū)別cumsum()
這篇文章主要介紹了prod()與cumprod()區(qū)別cumsum(),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-05-05