自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

漫話:如何給女朋友解釋為什么Java不支持多繼承?

開發(fā) 后端
面向?qū)ο蟮木幊陶Z言有三個重要的基本特性:封裝、繼承和多態(tài)。而很多人認為繼承是Java面向?qū)ο缶幊碳夹g的一塊基石。

 [[384385]]

要提到多繼承,首先要從繼承開始說起。

繼承

面向?qū)ο蟮木幊陶Z言有三個重要的基本特性:封裝、繼承和多態(tài)。而很多人認為繼承是Java面向?qū)ο缶幊碳夹g的一塊基石。

繼承就是子類繼承父類的特征和行為,使得子類對象(實例)具有父類的屬性和方法,或子類從父類繼承方法,使得子類具有父類相同的行為。

Java繼承是使用已存在的類的定義作為基礎建立新類的技術,新類的定義可以增加新的數(shù)據(jù)或新的功能,也可以用父類的功能,但不能選擇性地繼承父類。

加入,我們已經(jīng)定義了一個Car類,這個Car中包含了輪胎、發(fā)動機、底盤、方向盤等屬性,還具有行走、加油、開窗等行為。

而如果我們想要定義一輛Bus,想要復用這些屬性和行為,就可以通過繼承來實現(xiàn)。

通過使用繼承,我們使得Bus類和Car類之間存在了一定的關系,而我們通常稱呼Car是Bus的父類,Bus是Car的子類。

在Java中,使用extends關鍵字來實現(xiàn)繼承。

如上面Car與Bus,當寫繼承語句時,class Bus extends Car{ } 其中Bus類是子類,Car類是父類。

多繼承

上面我們提到的Bus和Car之間的關系其實是一種單繼承,指的是一個類只繼承自一個父類。

在軟件開發(fā)中,還有一種多繼承(多重繼承)的情況,顧名思義,就是一個類同時繼承自多個父類。

比如維基百科中關于多繼承舉了一個例子:

例如,可以創(chuàng)造一個“哺乳類動物”類別,擁有進食、繁殖等的功能;然后定義一個子類型“貓”,它可以從父類繼承上述功能,不需重新編寫程序,同時增加屬于自己的新功能,例如“追趕老鼠”。

但是,"貓"還可以作為"寵物"的子類,擁有一些寵物獨有的能力。

作為面向?qū)ο笳Z言,C++是支持多重繼承的。

但是,多年以來,多重繼承一直都是一個敏感的話題,反對者指它增加了程序的復雜性與含糊性。

 

Java不支持多繼承

很多人知道,Java是不支持多重繼承的,這里要提一下,這里的繼承特指的是使用extends關鍵字的這種繼承行為。

那么為什么Java不支持多重繼承呢?

關于這個問題,Java的創(chuàng)始人James Gosling曾經(jīng)回答過,他表示:

"Java之所以不支持一個類繼承多個類,主要是因為在設計之初我們聽取了來自C++和Objective-C登陣營的人的意見。因為多繼承會產(chǎn)生很多歧義問題。"

Gosling老人家提到的歧義問題,其實是C++因為支持多繼承之后帶來的菱形繼承問題。

假設我們有類B和類C,它們都繼承了相同的類A。另外我們還有類D,類D通過多重繼承機制繼承了類B和類C。

這時候,因為D同時繼承了B和C,并且B和C又同時繼承了A,那么,D中就會因為多重繼承,繼承到兩份來自A中的屬性和方法。

這時候,在使用D的時候,如果想要調(diào)用一個定義在A中的方法時,就會出現(xiàn)歧義。

因為這樣的繼承關系的形狀類似于菱形,因此這個問題被形象地稱為菱形繼承問題。

而C++為了解決菱形繼承問題,又引入了虛繼承。

因為支持多繼承,引入了菱形繼承問題,又因為要解決菱形繼承問題,引入了虛繼承。而經(jīng)過分析,人們發(fā)現(xiàn)我們其實真正想要使用多繼承的情況并不多。

所以,在 Java 中,不允許“實現(xiàn)多繼承”,即一個類不允許繼承多個父類。但是 Java 允許“聲明多繼承”,即一個類可以實現(xiàn)多個接口,一個接口也可以繼承多個父接口。由于接口只允許有方法聲明而不允許有方法實現(xiàn)(Java 8之前),這就避免了 C++ 中多繼承的歧義問題。

Java 8支持多繼承

Java不支持多繼承,但是是支持多實現(xiàn)的,也就是說,同一個類可以同時實現(xiàn)多個類。

我們知道,在Java 8以前,接口中是不能有方法的實現(xiàn)的。所以一個類同時實現(xiàn)多個接口的話,也不會出現(xiàn)C++中的歧義問題。因為所有方法都沒有方法體,真正的實現(xiàn)還是在子類中的。

那么問題來了。

Java 8中支持了默認函數(shù)(default method ),即接口中可以定義一個有方法體的方法了。

  1. public interface Pet { 
  2.  
  3.     public default void eat(){ 
  4.         System.out.println("Pet Is Eating"); 
  5.     } 

而又因為Java支持同時實現(xiàn)多個接口,這就相當于通過implements就可以從多個接口中繼承到多個方法了,這不就是變相支持了多繼承么。

那么,Java是怎么解決菱形繼承問題的呢?我們再定義一個哺乳動物接口,也定義一個eat方法。

  1. public interface Mammal { 
  2.  
  3.     public default void eat(){ 
  4.         System.out.println("Mammal Is Eating"); 
  5.     } 

然后定義一個Cat,讓他分別實現(xiàn)兩個接口:

  1. public class Cat implements Pet,Mammal { 
  2.  

這時候,編譯期會報錯:

  1. error: class Cat inherits unrelated defaults for eat() from types Mammal and Pet 

這時候,就要求Cat類中,必須重寫eat()方法。

  1. public class Cat implements Pet,Mammal { 
  2.     @Override 
  3.     public void eat() { 
  4.         System.out.println("Cat Is Eating"); 
  5.     } 

所以可以看到,Java并沒有幫我們解決多繼承的歧義問題,而是把這個問題留給開發(fā)人員,通過重寫方法的方式自己解決。

參考資料:

https://www.zhihu.com/question/24317891

關于作者:漫話編程,是一個通過漫畫+音頻的形式講解枯燥的編程知識的公眾號。致力于讓編程變得更有樂趣。

本文轉(zhuǎn)載自微信公眾號「漫話編程」,可以通過以下二維碼關注。轉(zhuǎn)載本文請聯(lián)系漫話編程公眾號。

 

責任編輯:武曉燕 來源: 漫話編程
相關推薦

2021-06-07 12:11:20

JavaRunning狀態(tài)

2020-03-16 14:08:59

線程熔斷限流

2021-05-10 19:58:06

MySQLUTF-8數(shù)據(jù)庫

2019-03-12 09:43:14

反向代理正向代理服務器

2021-09-14 12:00:11

VR字節(jié)跳動

2019-10-09 10:45:16

云計算Web互聯(lián)網(wǎng)

2021-04-26 14:00:43

Java 數(shù)據(jù)類型

2019-04-09 09:40:23

2019-12-23 10:26:02

3PC分布式2PC

2020-10-19 13:01:31

刪庫程序員思科

2019-04-26 14:46:18

GitGitHub局域網(wǎng)

2019-07-22 10:34:31

大案牘術大數(shù)據(jù)Big Data

2020-10-12 13:27:21

計算機瀏覽器電腦

2019-03-06 10:59:09

寬帶王者榮耀網(wǎng)絡

2019-04-19 09:48:53

樂觀鎖悲觀鎖數(shù)據(jù)庫

2022-05-09 10:28:42

SnapsUbuntuFlatpak

2020-12-28 12:37:36

緩存擊穿穿透

2020-03-23 12:57:20

撞庫脫庫洗庫

2019-08-13 10:34:26

鴻蒙OS跨平臺Linux內(nèi)核

2021-12-09 10:51:47

Go繼承
點贊
收藏

51CTO技術棧公眾號