繼承和多態(tài)

在OOP程序設(shè)計(jì)中,當(dāng)我們定義一個(gè)class的時(shí)候,可以從某個(gè)現(xiàn)有的class繼承,新的class稱為子類(Subclass),而被繼承的class稱為基類、父類或超類(Base class、Super class)。

比如,我們已經(jīng)編寫了一個(gè)名為Animal的class,有一個(gè)run()方法可以直接打印:

class Animal(object):

? ? ? ? ?def run(self):

? ? ? ? ? ? ? ? ? ? ?print 'Animal is running...'

當(dāng)我們需要編寫Dog和Cat類時(shí),就可以直接從Animal類繼承:

class Dog(Animal):

? ? ? ? ?pass

class Cat(Animal):

? ? ? ? ? pass

對于Dog來說,Animal就是它的父類,對于Animal來說,Dog就是它的子類。Cat和Dog類似。

繼承有什么好處?最大的好處是子類獲得了父類的全部功能。由于Animial實(shí)現(xiàn)了run()方法,因此,Dog和Cat作為它的子類,什么事也沒干,就自動(dòng)擁有了run()方法:

dog= Dog()

dog.run()

cat= Cat()

cat.run()

運(yùn)行結(jié)果如下:

Animal is running...

Animal is running...

當(dāng)然,也可以對子類增加一些方法,比如Dog類:

class Dog(Animal):

? ? ? ? ?def run(self):

? ? ? ? ? ? ? ? print 'Dog is running...'

? ? ? ? ?def eat(self):

? ? ? ? ? ? ? ? print'Eating meat...'

繼承的第二個(gè)好處需要我們對代碼做一點(diǎn)改進(jìn)。你看到了,無論是Dog還是Cat,它們r(jià)un()的時(shí)候,顯示的都是Animal is running...,符合邏輯的做法是分別顯示Dog is running...和Cat is running...,因此,對Dog和Cat類改進(jìn)如下:

classDog(Animal):defrun(self):print'Dog is running...'classCat(Animal):defrun(self):print'Cat is running...'

再次運(yùn)行,結(jié)果如下:

Dogisrunning...Catisrunning...

當(dāng)子類和父類都存在相同的run()方法時(shí),我們說,子類的run()覆蓋了父類的run(),在代碼運(yùn)行的時(shí)候,總是會(huì)調(diào)用子類的run()。這樣,我們就獲得了繼承的另一個(gè)好處:多態(tài)。

要理解什么是多態(tài),我們首先要對數(shù)據(jù)類型再作一點(diǎn)說明。當(dāng)我們定義一個(gè)class的時(shí)候,我們實(shí)際上就定義了一種數(shù)據(jù)類型。我們定義的數(shù)據(jù)類型和Python自帶的數(shù)據(jù)類型,比如str、list、dict沒什么兩樣:

a =list()# a是list類型b = Animal()# b是Animal類型c = Dog()# c是Dog類型

判斷一個(gè)變量是否是某個(gè)類型可以用isinstance()判斷:

>>>isinstance(a, list)True>>>isinstance(b, Animal)True>>>isinstance(c, Dog)True

看來a、b、c確實(shí)對應(yīng)著list、Animal、Dog這3種類型。

但是等等,試試:

>>>isinstance(c, Animal)True

看來c不僅僅是Dog,c還是Animal!

不過仔細(xì)想想,這是有道理的,因?yàn)镈og是從Animal繼承下來的,當(dāng)我們創(chuàng)建了一個(gè)Dog的實(shí)例c時(shí),我們認(rèn)為c的數(shù)據(jù)類型是Dog沒錯(cuò),但c同時(shí)也是Animal也沒錯(cuò),Dog本來就是Animal的一種!

所以,在繼承關(guān)系中,如果一個(gè)實(shí)例的數(shù)據(jù)類型是某個(gè)子類,那它的數(shù)據(jù)類型也可以被看做是父類。但是,反過來就不行:

>>>b = Animal()>>>isinstance(b, Dog)False

Dog可以看成Animal,但Animal不可以看成Dog。

要理解多態(tài)的好處,我們還需要再編寫一個(gè)函數(shù),這個(gè)函數(shù)接受一個(gè)Animal類型的變量:

defrun_twice(animal):animal.run()? ? animal.run()

當(dāng)我們傳入Animal的實(shí)例時(shí),run_twice()就打印出:

>>>run_twice(Animal())Animalisrunning...Animalisrunning...

當(dāng)我們傳入Dog的實(shí)例時(shí),run_twice()就打印出:

>>>run_twice(Dog())Dogisrunning...Dogisrunning...

當(dāng)我們傳入Cat的實(shí)例時(shí),run_twice()就打印出:

>>>run_twice(Cat())Catisrunning...Catisrunning...

看上去沒啥意思,但是仔細(xì)想想,現(xiàn)在,如果我們再定義一個(gè)Tortoise類型,也從Animal派生:

classTortoise(Animal):defrun(self):print'Tortoise is running slowly...'

當(dāng)我們調(diào)用run_twice()時(shí),傳入Tortoise的實(shí)例:

>>>run_twice(Tortoise())Tortoiseisrunning slowly...Tortoiseisrunning slowly...

你會(huì)發(fā)現(xiàn),新增一個(gè)Animal的子類,不必對run_twice()做任何修改,實(shí)際上,任何依賴Animal作為參數(shù)的函數(shù)或者方法都可以不加修改地正常運(yùn)行,原因就在于多態(tài)。

多態(tài)的好處就是,當(dāng)我們需要傳入Dog、Cat、Tortoise……時(shí),我們只需要接收Animal類型就可以了,因?yàn)镈og、Cat、Tortoise……都是Animal類型,然后,按照Animal類型進(jìn)行操作即可。由于Animal類型有run()方法,因此,傳入的任意類型,只要是Animal類或者子類,就會(huì)自動(dòng)調(diào)用實(shí)際類型的run()方法,這就是多態(tài)的意思:

對于一個(gè)變量,我們只需要知道它是Animal類型,無需確切地知道它的子類型,就可以放心地調(diào)用run()方法,而具體調(diào)用的run()方法是作用在Animal、Dog、Cat還是Tortoise對象上,由運(yùn)行時(shí)該對象的確切類型決定,這就是多態(tài)真正的威力:調(diào)用方只管調(diào)用,不管細(xì)節(jié),而當(dāng)我們新增一種Animal的子類時(shí),只要確保run()方法編寫正確,不用管原來的代碼是如何調(diào)用的。這就是著名的“開閉”原則:

對擴(kuò)展開放:允許新增Animal子類;

對修改封閉:不需要修改依賴Animal類型的run_twice()等函數(shù)。

繼承還可以一級(jí)一級(jí)地繼承下來,就好比從爺爺?shù)桨职帧⒃俚絻鹤舆@樣的關(guān)系。而任何類,最終都可以追溯到根類object,這些繼承關(guān)系看上去就像一顆倒著的樹。比如如下的繼承樹:

小結(jié)

繼承可以把父類的所有功能都直接拿過來,這樣就不必重零做起,子類只需要新增自己特有的方法,也可以把父類不適合的方法覆蓋重寫;

有了繼承,才能有多態(tài)。在調(diào)用類實(shí)例方法的時(shí)候,盡量把變量視作父類類型,這樣,所有子類類型都可以正常被接收;

舊的方式定義Python類允許不從object類繼承,但這種編程方式已經(jīng)嚴(yán)重不推薦使用。任何時(shí)候,如果沒有合適的類可以繼承,就繼承自object類。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,963評論 6 542
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,348評論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,083評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,706評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,442評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,802評論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,795評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,983評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,542評論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,287評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,486評論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,030評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,710評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,116評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,412評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,224評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,462評論 2 378

推薦閱讀更多精彩內(nèi)容

  • date: 2016-10-09 18:45:18 這里繼承和多態(tài)的概念與java的概念差不多。概念還是需要多次理...
    樸有天虹閱讀 273評論 0 0
  • 繼承和多態(tài): 在OOP程序設(shè)計(jì)中,當(dāng)我們定義一個(gè)class的時(shí)候,可以從某個(gè)現(xiàn)有的class繼承,新的class稱...
    黃大臻Dzreal閱讀 245評論 0 0
  • 在OPP程序設(shè)計(jì)中,當(dāng)我們定義一個(gè)class的時(shí)候,可以從某個(gè)現(xiàn)有的class繼承,新的class稱為子類。而被繼...
    界面大叔閱讀 261評論 0 0
  • 來自廖雪峰class(類)可以被繼承,新的class稱為子類(Subclass),被繼承的class是基類、父類或...
    Ji_uu閱讀 358評論 0 0
  • 繼承有什么好處?最大的好處是子類獲得了父類的全部功能。由于Animial實(shí)現(xiàn)了run()方法,因此,Dog和Cat...
    牛崽兒酷閱讀 389評論 0 0