多維度分析:抽象類和接口的區(qū)別

【編者按】本文作者是Sebastian Malaca,是面向?qū)ο缶幊痰目駸嵴?,不斷深化研究整潔代碼和高代碼質(zhì)量。本文中,作者通過(guò)多個(gè)方面深入剖析抽象類和接口的區(qū)別,并結(jié)合經(jīng)驗(yàn)供讀者借鑒學(xué)習(xí),本文系OneAPM工程師編譯整理。

在開發(fā)人員崗位面試時(shí),是否了解抽象類和接口之間的基本區(qū)別是一個(gè)很重要的考量因素。

顯而易見?

完全不是。筆者面試過(guò)很多人,通常問的第一個(gè)問題是關(guān)于接口和抽象類的區(qū)別。但實(shí)際上很少有程序員能給出正確的答案。

就這個(gè)問題來(lái)說(shuō),初級(jí)程序員可能都會(huì)清楚之間的區(qū)別,可能也并不一定理解其背后的原因,但其結(jié)構(gòu)上的差異,特別是針對(duì)特定語(yǔ)言(幾乎和所有的面向?qū)ο蟮恼Z(yǔ)言一樣)應(yīng)該深入了解。

同時(shí),筆者也發(fā)現(xiàn)其他職位候選人(有時(shí)甚至是高級(jí)職位)竟然也不知道這之間的差異,或者只知道的一個(gè)或幾個(gè)。

如果只是需要了解這些內(nèi)容那并不難,但這些都是面向?qū)ο蟮幕A(chǔ)知識(shí),因此想要設(shè)計(jì)良好的代碼必須對(duì)其有一個(gè)深入的認(rèn)識(shí)。

下面將詳細(xì)介紹這些基礎(chǔ)知識(shí)。

繼承

下面將從眾所周知的接口和抽象類的區(qū)別開始。這種差異是關(guān)于繼承的,任何類都可以實(shí)現(xiàn)多個(gè)接口,但是只能擴(kuò)展一個(gè)類,也只能有一個(gè)父類。

多個(gè)類擴(kuò)展是一個(gè)語(yǔ)言特性,它存在于一些面向?qū)ο蟮恼Z(yǔ)言。為什么呢?因?yàn)樗鼛?lái)的問題往往多于價(jià)值。

當(dāng)一個(gè)類有許多父類時(shí),有一個(gè)情況就是完全相同的方法會(huì)聲明多個(gè),因此必須顯式地「告知」究竟需要的是哪一個(gè)。

這樣的代碼通常難以維護(hù),因?yàn)閷?duì)其進(jìn)行的任何修改或者重構(gòu)都必須小心地檢查。另一方面,如果一個(gè)類需要擴(kuò)展(至少)兩個(gè)擁有相同方法的類,那么 DRY 規(guī)則顯然會(huì)被破壞(因此需要從別處下手),或者說(shuō)會(huì)干擾到 Single Responsibility Principle (SAP)。

「如果多個(gè)類的繼承如此糟糕,為什么它可以實(shí)現(xiàn)許多接口呢?」——如果這樣的問題在你的腦海盤旋,我不得不承認(rèn)這是一個(gè)絕妙的問題。

然而,答案很簡(jiǎn)單。每一個(gè)接口都是基于函數(shù)而不是一個(gè)類去實(shí)現(xiàn)。所以,即使實(shí)現(xiàn)十個(gè)不同的接口,每個(gè)包含相同的方法聲明,內(nèi)部也不會(huì)發(fā)生沖突。接口保證了方法的存在,而不是去說(shuō)明方法的實(shí)現(xiàn),這意味著,只要不違反 SRP ,你完全可以實(shí)現(xiàn)多個(gè)接口。

方法的可見度

接口中的所有方法都是 Public 的,但對(duì)于抽象類的聲明并沒有這樣的規(guī)則,當(dāng)然不能是 Private 。為什么不能 Private?因?yàn)橐粋€(gè)抽象方法需要在子類中實(shí)現(xiàn),但 Private 無(wú)法訪問子類,因此不抽象類不可能存在 Private 屬性。

接著回歸主題。正如上文寫道的,接口是一個(gè)函數(shù)的保證,你可以把它當(dāng)作使用接口的類和實(shí)現(xiàn)這個(gè)接口的類之間的一個(gè)合約——保證一個(gè)特定類將實(shí)現(xiàn)所有聲明的方法。這也是為什么這些方法必須是 Public 的原因。因?yàn)楸粐?yán)格的限制到了實(shí)現(xiàn)上,所以其他一切都不成問題。

然而,當(dāng)涉及到抽象類時(shí)并非這樣。我們總是可以有不同的類組,除了這幾方面基本上不同以外,其他地方幾乎一樣,類體的公共方法也是非常相似的。在這種情況下,可以創(chuàng)建 Protected 方法來(lái)保持類之間的差異。Template Method 就是一個(gè)很典型的例子。

聲明和定義

接口只能包含方法聲明,而抽象類還可以包含方法的定義。

接口的重點(diǎn)在于提供特定函數(shù),而抽象類還在于子類實(shí)現(xiàn)的相似性,不僅僅是其中的函數(shù)。

常量

接口和抽象類中都可以定義常量。這是因?yàn)檫@些值不依賴于特定對(duì)象,對(duì)它們來(lái)說(shuō)都是相同的。

屬性

抽象類可以包含屬性,但接口卻不能。原因與聲明和定義是一樣的。

總結(jié)

除了說(shuō)明差異,筆者也試圖解釋它產(chǎn)生的原因。這不僅是因?yàn)槿藗儼l(fā)明某個(gè)語(yǔ)言時(shí)的突發(fā)奇想,而是源于語(yǔ)言背后所支撐的理念。

原文鏈接:Differences Between Abstract Class and Interface

OneAPM 是應(yīng)用性能管理領(lǐng)域的新興領(lǐng)軍企業(yè),能幫助企業(yè)用戶和開發(fā)者輕松實(shí)現(xiàn):緩慢的程序代碼和 SQL 語(yǔ)句的實(shí)時(shí)抓取。想閱讀更多技術(shù)文章,請(qǐng)?jiān)L問OneAPM 官方博客

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

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

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法,類相關(guān)的語(yǔ)法,內(nèi)部類的語(yǔ)法,繼承相關(guān)的語(yǔ)法,異常的語(yǔ)法,線程的語(yǔ)...
    子非魚_t_閱讀 31,721評(píng)論 18 399
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,785評(píng)論 18 139
  • 先來(lái)看看抽象類和接口的定義: 抽象類(abstract class): 使用abstract修飾符修飾的類。官方點(diǎn)...
    IT廢柴閱讀 53,836評(píng)論 7 45
  • 一個(gè)WordPress主題至少包含以下兩個(gè)文件: style.css index.php 好,現(xiàn)在就開始我們的Wo...
    弈梵說(shuō)閱讀 903評(píng)論 0 0
  • 既然它是一灘死水, 那么請(qǐng)你, 不要隨便亂丟石頭。 死寂的水面, ...
    孤羅閱讀 489評(píng)論 0 1