摘要:更多關(guān)于接口和抽象類的概念知識,可自行查閱相關(guān)文檔。抽象類及其作用抽象類,顧名思義,即類的抽象。而接口是對行為的抽象,表達的是的關(guān)系。相比于抽象類,接口能最大限度的減少對外暴露的接口,并隱藏細節(jié),更符合這一原則。
作者:伯特
出處:github.com/ruicbAndroid/LoulanPlan
聲明:本文出自伯特的《樓蘭計劃》,轉(zhuǎn)載務必注明作者及出處。
本文旨在討論抽象類和接口的作用、實例及使用場景,都是我的理解和總結(jié)。更多關(guān)于接口和抽象類的概念知識,可自行查閱相關(guān)文檔。
1. 抽象類及其作用抽象類,顧名思義,即類的抽象。
在介紹面向?qū)ο蟾拍顣r,我們知道類是客觀事物的抽象,而抽象類又是類的進一步抽象,該怎么理解呢?
舉個例子,我們定義若干個類 class BMW、class Benz、class Audi,分別對客觀事物“寶馬”、“奔馳”、“奧迪”三種汽車進行抽象,包含相關(guān)屬性和行為(即方法)。但是我們知道,汽車都有通用的屬性和行為,比如品牌、發(fā)動機、方向盤、輪胎等屬性,前進、后退、轉(zhuǎn)彎等行為,所以我們可以在寶馬、奔馳等汽車之上,進一步抽象出“汽車”類 abstract class Car,包含通用的特性(屬性和方法)。讓 BMW、Benz、Audi 等繼承抽象類 extends Car,便擁有了汽車的通用特性,然后在抽象類基礎上定義各自的特殊屬性及方法。
這里的 abstract class Car 即抽象類,可以看出,抽象類是用來捕捉子類的通用特性的,包括屬性及行為。
2. 接口及其作用下面我們來看看接口,假使我研發(fā)出來一臺會飛的汽車“伯特萊斯”(Bote-Royce),在程序中定義如下:
class BoteRoyce extends Car { //...省略通用特性 /** * 可以飛 */ void fly() { System.out.println("假裝會飛~"); } }
看起來沒問題:
BoteRoyce extends Car:表達這是一輛汽車;
fly() 方法:體現(xiàn)這車可以飛。
但是,隨著技術(shù)發(fā)展,出現(xiàn)了眾多可以制造飛行汽車的廠商,難道每一個可以飛的汽車都去定義一個 fly() 方法?
心想這還不簡單,在抽象類 Car 中定義一個抽象方法 abstract void fly() 讓子類去實現(xiàn),不就可以了嗎?
No No No... 正如不是所有牛奶都叫特侖蘇一樣,不是所有汽車都會飛,飛行功能不是汽車的通用特性。將 fly() 方法定義在 Car 中,顯然違背了“抽象類用來捕捉子類的通用特性”這一原則。
在這種場景下,解決方案之一就是使用接口,如下:
/** * 飛行器接口 */ public interface Aircraft { //定義抽象方法 void fly(); }
類 BoteRoyce 的定義修改如下:
/* * 實現(xiàn) Aircraft 接口,表示具備飛行器能力 */ class BoteRoyce extends Car implements Aircraft { /** * 覆寫接口方法,實現(xiàn)飛行能力 */ @Override void fly() { System.out.println("假裝會飛~"); } }
再有其他品牌的飛行汽車,都可以通過 extends Car implements Aircraft 實現(xiàn)飛行能力。
上述定義的 interface Aircraft 即為接口,我們通常使用接口對行為進行抽象。
3. 接口和抽象類的區(qū)別關(guān)于二者的區(qū)別,可以結(jié)合前面的例子,來加深理解。
抽象類是對類本質(zhì)的抽象,表達的是 is a 的關(guān)系,比如:BMW is a Car。抽象類包含并實現(xiàn)子類的通用特性,將子類存在差異化的特性進行抽象,交由子類去實現(xiàn)。
而接口是對行為的抽象,表達的是 like a 的關(guān)系。比如:Bote-Royce like a Aircraft(像飛行器一樣可以飛),但其本質(zhì)上 is a Car。接口的核心是定義行為,即實現(xiàn)類可以做什么,至于實現(xiàn)類主體是誰、是如何實現(xiàn)的,接口并不關(guān)心。
4. 接口與抽象類的使用場景熟悉 Java 的同學可能會質(zhì)疑,上述關(guān)于接口的使用,完全可以通過再次抽象 Car 去實現(xiàn):
/** * 會飛的汽車 */ abstract class FlyCar extends Car { //定義抽象方法 public abstract void fly(); }
普通的汽車依然 extends Car,可以飛行的汽車 extends FlyCar 即可:
/* * 繼承 FlyCar,表示是可以飛行的汽車 */ class BoteRoyce extends FlyCar { /** * 覆寫抽象方法,實現(xiàn)飛行能力 */ @Override public void fly() { System.out.println("假裝會飛~"); } }
如果你也這么想,表示你 get 到了抽象類的點。不過話說回來,這樣的話接口豈不是沒有存在的意義了?
當然不是了。就 BoteRoyce 而言,如果你關(guān)心的是“飛行汽車”這個整體,那么定義抽象類 FlyCar 是個不錯的選擇;如果你關(guān)心的是汽車具備“飛行”的行為,那不妨繼續(xù)沿用前面使用 Aircraft 接口的方案。
這一點與設計模式中六大原則之一的“里氏替換原則”不謀而合,該原則指出:所有引用基類(抽象類或接口)的地方必須能透明地使用其子類的對象。也就是說,當你遵循該原則時,你必須要考慮你關(guān)心的是“飛行汽車”實體,還是“飛行”行為,并將其作為基類,從而決定程序所能接受的子類對象。
同時,“接口隔離原則”指導我們,一個類對另一個類的依賴應該建立在最小的接口上。相比于抽象類 FlyCar,接口 Aircraft 能最大限度的減少對外暴露的接口,并隱藏細節(jié),更符合這一原則。
所以說啊,面向?qū)ο笾皇侵笇覀兙幊痰乃枷耄菞l條框框。在實際開發(fā)中,具體使用抽象類還是接口,并沒有絕對限制,而是取決于你的業(yè)務場景和架構(gòu)設計。
5. 總結(jié)好了,本次關(guān)于接口與抽象類的總結(jié)就到這兒,你徹底弄懂了嗎?下期分享再見~
歡迎關(guān)注我的公眾號“伯特說”:
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/72627.html
摘要:事件冒泡一個簡單,但是坑了我無數(shù)回的知識點與的交互通過事件來實現(xiàn)。而瀏覽器的事件流是一個非常重要的概念。不去討論那些古老的瀏覽器有事件捕獲與事件冒泡的爭議,只需要知道在中規(guī)定的事件流包括了三個部分,事件捕獲階段處于目標階段事件冒泡階段。 打算封裝一個彈窗組件,做的時候忘記了考慮事件冒泡的因素,結(jié)果被坑得不要不要的。為了解決自己的問題,去查閱了不少資料,把事件流相關(guān)的知識都給總結(jié)一下。 ...
摘要:事件冒泡一個簡單,但是坑了我無數(shù)回的知識點與的交互通過事件來實現(xiàn)。而瀏覽器的事件流是一個非常重要的概念。不去討論那些古老的瀏覽器有事件捕獲與事件冒泡的爭議,只需要知道在中規(guī)定的事件流包括了三個部分,事件捕獲階段處于目標階段事件冒泡階段。 打算封裝一個彈窗組件,做的時候忘記了考慮事件冒泡的因素,結(jié)果被坑得不要不要的。為了解決自己的問題,去查閱了不少資料,把事件流相關(guān)的知識都給總結(jié)一下。 ...
閱讀 1686·2021-11-23 09:51
閱讀 2696·2021-11-22 09:34
閱讀 1330·2021-10-14 09:43
閱讀 3672·2021-09-08 09:36
閱讀 3217·2019-08-30 12:57
閱讀 2039·2019-08-30 12:44
閱讀 2528·2019-08-29 17:15
閱讀 3024·2019-08-29 16:08