淺析Python中的多重繼承
繼承是面向?qū)ο缶幊痰囊粋€(gè)重要的方式,因?yàn)橥ㄟ^繼承,子類就可以擴(kuò)展父類的功能。
回憶一下Animal類層次的設(shè)計(jì),假設(shè)我們要實(shí)現(xiàn)以下4種動(dòng)物:
- Dog - 狗狗;
- Bat - 蝙蝠;
- Parrot - 鸚鵡;
- Ostrich - 鴕鳥。
如果按照哺乳動(dòng)物和鳥類歸類,我們可以設(shè)計(jì)出這樣的類的層次:

但是如果按照“能跑”和“能飛”來歸類,我們就應(yīng)該設(shè)計(jì)出這樣的類的層次:

如果要把上面的兩種分類都包含進(jìn)來,我們就得設(shè)計(jì)更多的層次:
- 哺乳類:能跑的哺乳類,能飛的哺乳類;
- 鳥類:能跑的鳥類,能飛的鳥類。
這么一來,類的層次就復(fù)雜了:

如果要再增加“寵物類”和“非寵物類”,這么搞下去,類的數(shù)量會(huì)呈指數(shù)增長,很明顯這樣設(shè)計(jì)是不行的。
正確的做法是采用多重繼承。首先,主要的類層次仍按照哺乳類和鳥類設(shè)計(jì):
class Animal(object): pass # 大類: class Mammal(Animal): pass class Bird(Animal): pass # 各種動(dòng)物: class Dog(Mammal): pass class Bat(Mammal): pass class Parrot(Bird): pass class Ostrich(Bird): pass
現(xiàn)在,我們要給動(dòng)物再加上Runnable和Flyable的功能,只需要先定義好Runnable和Flyable的類:
class Runnable(object):
def run(self):
print('Running...')
class Flyable(object):
def fly(self):
print('Flying...')
對于需要Runnable功能的動(dòng)物,就多繼承一個(gè)Runnable,例如Dog:
class Dog(Mammal, Runnable): pass
對于需要Flyable功能的動(dòng)物,就多繼承一個(gè)Flyable,例如Bat:
class Bat(Mammal, Flyable): pass
通過多重繼承,一個(gè)子類就可以同時(shí)獲得多個(gè)父類的所有功能。
Mixin
在設(shè)計(jì)類的繼承關(guān)系時(shí),通常,主線都是單一繼承下來的,例如,Ostrich繼承自Bird。但是,如果需要“混入”額外的功能,通過多重繼承就可以實(shí)現(xiàn),比如,讓Ostrich除了繼承自Bird外,再同時(shí)繼承Runnable。這種設(shè)計(jì)通常稱之為Mixin。
為了更好地看出繼承關(guān)系,我們把Runnable和Flyable改為RunnableMixin和FlyableMixin。類似的,你還可以定義出肉食動(dòng)物CarnivorousMixin和植食動(dòng)物HerbivoresMixin,讓某個(gè)動(dòng)物同時(shí)擁有好幾個(gè)Mixin:
class Dog(Mammal, RunnableMixin, CarnivorousMixin): pass
Mixin的目的就是給一個(gè)類增加多個(gè)功能,這樣,在設(shè)計(jì)類的時(shí)候,我們優(yōu)先考慮通過多重繼承來組合多個(gè)Mixin的功能,而不是設(shè)計(jì)多層次的復(fù)雜的繼承關(guān)系。
Python自帶的很多庫也使用了Mixin。舉個(gè)例子,Python自帶了TCPServer和UDPServer這兩類網(wǎng)絡(luò)服務(wù),而要同時(shí)服務(wù)多個(gè)用戶就必須使用多進(jìn)程或多線程模型,這兩種模型由ForkingMixin和ThreadingMixin提供。通過組合,我們就可以創(chuàng)造出合適的服務(wù)來。
比如,編寫一個(gè)多進(jìn)程模式的TCP服務(wù),定義如下:
class MyTCPServer(TCPServer, ForkingMixin): pass
編寫一個(gè)多線程模式的UDP服務(wù),定義如下:
class MyUDPServer(UDPServer, ThreadingMixin): pass
如果你打算搞一個(gè)更先進(jìn)的協(xié)程模型,可以編寫一個(gè)CoroutineMixin:
class MyTCPServer(TCPServer, CoroutineMixin): pass
這樣一來,我們不需要復(fù)雜而龐大的繼承鏈,只要選擇組合不同的類的功能,就可以快速構(gòu)造出所需的子類。
小結(jié)
由于Python允許使用多重繼承,因此,Mixin就是一種常見的設(shè)計(jì)。
只允許單一繼承的語言(如Java)不能使用Mixin的設(shè)計(jì)。
- Python類的多重繼承問題深入分析
- Python 的類、繼承和多態(tài)詳解
- 在python里面運(yùn)用多繼承方法詳解
- python中的多重繼承實(shí)例講解
- Python 多線程其他屬性以及繼承Thread類詳解
- Python多重繼承的方法解析執(zhí)行順序?qū)嵗治?/a>
- Python中的單繼承與多繼承實(shí)例分析
- python多重繼承實(shí)例
- Python類的繼承與多態(tài)詳細(xì)介紹
- Python面向?qū)ο缶幊讨^承與多態(tài)詳解
- Python實(shí)現(xiàn)帶參數(shù)與不帶參數(shù)的多重繼承示例
- Python中多層繼承的實(shí)現(xiàn)示例
相關(guān)文章
python利用Tesseract識別驗(yàn)證碼的方法示例
無論是是自動(dòng)化登錄還是爬蟲,總繞不開驗(yàn)證碼,這次就來談?wù)刾ython中光學(xué)識別驗(yàn)證碼模塊Tesseract,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01
解決python通過cx_Oracle模塊連接Oracle亂碼的問題
今天小編就為大家分享一篇解決python通過cx_Oracle模塊連接Oracle亂碼的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-10-10
Python 使用dict實(shí)現(xiàn)switch的操作
這篇文章主要介紹了Python 使用dict實(shí)現(xiàn)switch的操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-04-04
python和flask中返回JSON數(shù)據(jù)的方法
下面小編就為大家整理了一篇python和flask中返回JSON數(shù)據(jù)的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-03-03
python使用JSON模塊進(jìn)行數(shù)據(jù)處理(編碼解碼)
這篇文章主要為大家介紹了python使用JSON模塊進(jìn)行數(shù)據(jù)處理編碼解碼的使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
如何使用python實(shí)現(xiàn)模擬鼠標(biāo)點(diǎn)擊
這篇文章主要介紹了如何使用python實(shí)現(xiàn)模擬鼠標(biāo)點(diǎn)擊,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01
python實(shí)現(xiàn)輸出一個(gè)序列的所有子序列示例
今天小編就為大家分享一篇python實(shí)現(xiàn)輸出一個(gè)序列的所有子序列示例,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-11-11

