摘要:本文中,作者通過多個方面深入剖析抽象類和接口的區別,并結合經驗供讀者借鑒學習,本文系工程師編譯整理。在開發人員崗位面試時,是否了解抽象類和接口之間的基本區別是一個很重要的考量因素。
【編者按】本文作者是Sebastian Malaca,是面向對象編程的狂熱者,不斷深化研究整潔代碼和高代碼質量。本文中,作者通過多個方面深入剖析抽象類和接口的區別,并結合經驗供讀者借鑒學習,本文系 OneAPM 工程師編譯整理。
在開發人員崗位面試時,是否了解抽象類和接口之間的基本區別是一個很重要的考量因素。
顯而易見?
完全不是。筆者面試過很多人,通常問的第一個問題是關于接口和抽象類的區別。但實際上很少有程序員能給出正確的答案。
就這個問題來說,初級程序員可能都會清楚之間的區別,可能也并不一定理解其背后的原因,但其結構上的差異,特別是針對特定語言(幾乎和所有的面向對象的語言一樣)應該深入了解。
同時,筆者也發現其他職位候選人(有時甚至是高級職位)竟然也不知道這之間的差異,或者只知道的一個或幾個。
如果只是需要了解這些內容那并不難,但這些都是面向對象的基礎知識,因此想要設計良好的代碼必須對其有一個深入的認識。
下面將詳細介紹這些基礎知識。
繼承
下面將從眾所周知的接口和抽象類的區別開始。這種差異是關于繼承的,任何類都可以實現多個接口,但是只能擴展一個類,也只能有一個父類。
多個類擴展是一個語言特性,它存在于一些面向對象的語言。為什么呢?因為它帶來的問題往往多于價值。
當一個類有許多父類時,有一個情況就是完全相同的方法會聲明多個,因此必須顯式地「告知」究竟需要的是哪一個。
這樣的代碼通常難以維護,因為對其進行的任何修改或者重構都必須小心地檢查。另一方面,如果一個類需要擴展(至少)兩個擁有相同方法的類,那么 DRY 規則顯然會被破壞(因此需要從別處下手),或者說會干擾到 Single Responsibility Principle (SAP)。
「如果多個類的繼承如此糟糕,為什么它可以實現許多接口呢?」——如果這樣的問題在你的腦海盤旋,我不得不承認這是一個絕妙的問題。
然而,答案很簡單。每一個接口都是基于函數而不是一個類去實現。所以,即使實現十個不同的接口,每個包含相同的方法聲明,內部也不會發生沖突。接口保證了方法的存在,而不是去說明方法的實現,這意味著,只要不違反 SRP ,你完全可以實現多個接口。
方法的可見度
接口中的所有方法都是 Public 的,但對于抽象類的聲明并沒有這樣的規則,當然不能是 Private 。為什么不能 Private?因為一個抽象方法需要在子類中實現,但 Private 無法訪問子類,因此不抽象類不可能存在 Private 屬性。
接著回歸主題。正如上文寫道的,接口是一個函數的保證,你可以把它當作使用接口的類和實現這個接口的類之間的一個合約——保證一個特定類將實現所有聲明的方法。這也是為什么這些方法必須是 Public 的原因。因為被嚴格的限制到了實現上,所以其他一切都不成問題。
然而,當涉及到抽象類時并非這樣。我們總是可以有不同的類組,除了這幾方面基本上不同以外,其他地方幾乎一樣,類體的公共方法也是非常相似的。在這種情況下,可以創建 Protected 方法來保持類之間的差異。Template Method 就是一個很典型的例子。
聲明和定義
接口只能包含方法聲明,而抽象類還可以包含方法的定義。
接口的重點在于提供特定函數,而抽象類還在于子類實現的相似性,不僅僅是其中的函數。
常量
接口和抽象類中都可以定義常量。這是因為這些值不依賴于特定對象,對它們來說都是相同的。
屬性
抽象類可以包含屬性,但接口卻不能。原因與聲明和定義是一樣的。
總結
除了說明差異,筆者也試圖解釋它產生的原因。這不僅是因為人們發明某個語言時的突發奇想,而是源于語言背后所支撐的理念。
原文鏈接:Differences Between Abstract Class and Interface
OneAPM 是應用性能管理領域的新興領軍企業,能幫助企業用戶和開發者輕松實現:緩慢的程序代碼和 SQL 語句的實時抓取。想閱讀更多技術文章,請訪問 OneAPM 官方博客。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/64522.html
摘要:橋接模式和裝飾模式的區別設計模式裝飾模式橋接模式和裝飾模式都是通過將繼承關系轉換為關聯關系從而減少系統中類的數量,降低系統的耦合性。裝飾器模式支持多層裝飾,通過不同的組合可以實現不同的行為。 產生橋接模式的動機: 假設這樣一種情況:我們有大中小型號的毛筆,有紅藍黑三種顏料。如果需要不同顏色,不同型號的毛筆有如下兩種設計方法: 為每一種型號的毛筆都提供三種顏料的版本。 將毛筆和顏料分開...
摘要:抽象方法沒有方法體抽象方法權限修飾符不能為抽象方法的目的就是為了讓子類繼承重寫的,所以抽象方法不能私有,不能修飾。另外,添加默認方法不會影響函數式接口的使用。 原文:抽象類和接口的區別已經變了 隨著JDK的不斷迭代,抽象類和接口的區別已經有了些許改變,你是否還停留在JDK 7 的答案呢? 定義 抽象類定義通過 abstract class public abstract class A...
摘要:如果一個非抽象類遵循了某個接口,就必須實現該接口中的所有方法。抽象類是對整個類整體進行抽象,包括屬性行為,但是接口卻是對類局部行為進行抽象。因此最好的解決辦法是單獨將報警設計為一個接口,包含行為設計為單獨的一個抽象類,包含和兩種行為。 一、抽象類 二、接口 三、抽象類和接口的區別 一、抽象類 在了解抽象類之前,先來了解一下抽象方法。抽象方法是一種特殊的方法:它只有聲明,而沒有具體的實現...
閱讀 2953·2021-11-23 09:51
閱讀 1006·2021-09-26 09:55
閱讀 3935·2021-09-22 14:58
閱讀 1468·2021-09-08 09:35
閱讀 1078·2021-08-26 14:16
閱讀 882·2019-08-23 18:17
閱讀 2054·2019-08-23 16:45
閱讀 700·2019-08-23 15:55