回答這個(gè)我找到比較有意思的回答,咱們看漫畫來了解:
要提到多繼承,首先要從繼承開始說起。
繼承
面向?qū)ο蟮木幊陶Z言有三個(gè)重要的基本特性:封裝、繼承和多態(tài)。而很多人認(rèn)為繼承是Java面向?qū)ο缶幊碳夹g(shù)的一塊基石。
繼承就是子類繼承父類的特征和行為,使得子類對(duì)象(實(shí)例)具有父類的屬性和方法,或子類從父類繼承方法,使得子類具有父類相同的行為。
Java繼承是使用已存在的類的定義作為基礎(chǔ)建立新類的技術(shù),新類的定義可以增加新的數(shù)據(jù)或新的功能,也可以用父類的功能,但不能選擇性地繼承父類。
加入,我們已經(jīng)定義了一個(gè)Car類,這個(gè)Car中包含了輪胎、發(fā)動(dòng)機(jī)、底盤、方向盤等屬性,還具有行走、加油、開窗等行為。
而如果我們想要定義一輛Bus,想要復(fù)用這些屬性和行為,就可以通過繼承來實(shí)現(xiàn)。
通過使用繼承,我們使得Bus類和Car類之間存在了一定的關(guān)系,而我們通常稱呼Car是Bus的父類,Bus是Car的子類。
在Java中,使用extends關(guān)鍵字來實(shí)現(xiàn)繼承。
如上面Car與Bus,當(dāng)寫繼承語句時(shí),class Bus extends Car{ } 其中Bus類是子類,Car類是父類。
多繼承
上面我們提到的Bus和Car之間的關(guān)系其實(shí)是一種單繼承,指的是一個(gè)類只繼承自一個(gè)父類。
在軟件開發(fā)中,還有一種多繼承(多重繼承)的情況,顧名思義,就是一個(gè)類同時(shí)繼承自多個(gè)父類。
比如維基百科中關(guān)于多繼承舉了一個(gè)例子:
例如,可以創(chuàng)造一個(gè)“哺乳類動(dòng)物”類別,擁有進(jìn)食、繁殖等的功能;然后定義一個(gè)子類型“貓”,它可以從父類繼承上述功能,不需重新編寫程序,同時(shí)增加屬于自己的新功能,例如“追趕老鼠”。
但是,"貓"還可以作為"寵物"的子類,擁有一些寵物獨(dú)有的能力。
作為面向?qū)ο笳Z言,C++是支持多重繼承的。
但是,多年以來,多重繼承一直都是一個(gè)敏感的話題,反對(duì)者指它增加了程序的復(fù)雜性與含糊性。
Java不支持多繼承
很多人知道,Java是不支持多重繼承的,這里要提一下,這里的繼承特指的是使用extends關(guān)鍵字的這種繼承行為。
那么為什么Java不支持多重繼承呢?
關(guān)于這個(gè)問題,Java的創(chuàng)始人James Gosling曾經(jīng)回答過,他表示:
"Java之所以不支持一個(gè)類繼承多個(gè)類,主要是因?yàn)樵谠O(shè)計(jì)之初我們聽取了來自C++和Objective-C登陣營的人的意見。因?yàn)槎嗬^承會(huì)產(chǎn)生很多歧義問題。"
Gosling老人家提到的歧義問題,其實(shí)是C++因?yàn)橹С侄嗬^承之后帶來的菱形繼承問題。
假設(shè)我們有類B和類C,它們都繼承了相同的類A。另外我們還有類D,類D通過多重繼承機(jī)制繼承了類B和類C。
這時(shí)候,因?yàn)镈同時(shí)繼承了B和C,并且B和C又同時(shí)繼承了A,那么,D中就會(huì)因?yàn)槎嘀乩^承,繼承到兩份來自A中的屬性和方法。
這時(shí)候,在使用D的時(shí)候,如果想要調(diào)用一個(gè)定義在A中的方法時(shí),就會(huì)出現(xiàn)歧義。
因?yàn)檫@樣的繼承關(guān)系的形狀類似于菱形,因此這個(gè)問題被形象地稱為菱形繼承問題。
而C++為了解決菱形繼承問題,又引入了虛繼承。
因?yàn)橹С侄嗬^承,引入了菱形繼承問題,又因?yàn)橐鉀Q菱形繼承問題,引入了虛繼承。而經(jīng)過分析,人們發(fā)現(xiàn)我們其實(shí)真正想要使用多繼承的情況并不多。
所以,在 Java 中,不允許“實(shí)現(xiàn)多繼承”,即一個(gè)類不允許繼承多個(gè)父類。但是 Java 允許“聲明多繼承”,即一個(gè)類可以實(shí)現(xiàn)多個(gè)接口,一個(gè)接口也可以繼承多個(gè)父接口。由于接口只允許有方法聲明而不允許有方法實(shí)現(xiàn)(Java 8之前),這就避免了 C++ 中多繼承的歧義問題。
Java 8支持多繼承
Java不支持多繼承,但是是支持多實(shí)現(xiàn)的,也就是說,同一個(gè)類可以同時(shí)實(shí)現(xiàn)多個(gè)類。我們知道,在Java 8以前,接口中是不能有方法的實(shí)現(xiàn)的。所以一個(gè)類同時(shí)實(shí)現(xiàn)多個(gè)接口的話,也不會(huì)出現(xiàn)C++中的歧義問題。因?yàn)樗蟹椒ǘ紱]有方法體,真正的實(shí)現(xiàn)還是在子類中的。
那么問題來了,Java 8中支持了默認(rèn)函數(shù)(default method ),即接口中可以定義一個(gè)有方法體的方法了。
- public interface Pet {
- public default void eat(){
- System.out.println("Pet Is Eating");
- }
- }
而又因?yàn)镴ava支持同時(shí)實(shí)現(xiàn)多個(gè)接口,這就相當(dāng)于通過implements就可以從多個(gè)接口中繼承到多個(gè)方法了,這不就是變相支持了多繼承么。
那么,Java是怎么解決菱形繼承問題的呢?我們?cè)俣x一個(gè)哺乳動(dòng)物接口,也定義一個(gè)eat方法。
- public interface Mammal {
- public default void eat(){
- System.out.println("Mammal Is Eating");
- }
- }
然后定義一個(gè)Cat,讓他分別實(shí)現(xiàn)兩個(gè)接口:
- public class Cat implements Pet,Mammal {
- }
這時(shí)候,編譯期會(huì)報(bào)錯(cuò):
- error: class Cat inherits unrelated defaults for eat() from types Mammal and Pet
這時(shí)候,就要求Cat類中,必須重寫eat()方法。
- public class Cat implements Pet,Mammal {
- @Override
- public void eat() {
- System.out.println("Cat Is Eating");
- }
- }
所以可以看到,Java并沒有幫我們解決多繼承的歧義問題,而是把這個(gè)問題留給開發(fā)人員,通過重寫方法的方式自己解決。
更多關(guān)于“Java培訓(xùn)”的問題,歡迎咨詢千鋒教育在線名師。千鋒已有十余年的培訓(xùn)經(jīng)驗(yàn),課程大綱更科學(xué)更專業(yè),有針對(duì)零基礎(chǔ)的就業(yè)班,有針對(duì)想提升技術(shù)的好程序員班,高品質(zhì)課程助理你實(shí)現(xiàn)java程序員夢(mèng)想。