01、抽象類和接口的區(qū)別
來看網(wǎng)絡(luò)上對接口的一番解釋:
接口(英文:Interface),在 Java 編程語言中是一個抽象類型,是抽象方法的集合。一個類通過繼承接口的方式,從而來繼承接口的抽象方法。
兄弟們,你們怎么看,這段解釋把我繞得暈乎乎的,好像喝過一斤二鍋頭。到底是解釋抽象類呢還是接口呢?傻傻分不清楚。
搞不清楚要用抽象類還是接口,就先來看看兩者之間的區(qū)別。來,抽象類和接口,你倆過來比比身高。
抽象類中的方法可以有方法體,能實現(xiàn)方法具體要實現(xiàn)的功能,但是接口中的方法不行,沒有方法體。
抽象類中的成員變量可以是各種類型的,而接口中的成員變量只能是 public static final 類型的,并且是隱式的,缺省的。
接口中不能含有靜態(tài)代碼塊以及靜態(tài)方法(用 static 修飾的方法),而抽象類是可以有靜態(tài)代碼塊和靜態(tài)方法的。
一個類只能繼承一個抽象類,而一個類卻可以實現(xiàn)多個接口。
02、進一步剖析接口和抽象類
好像知道了兩者之間的區(qū)別,但印象還是有些模糊。沒關(guān)系,我們進一步深入。
抽象類
抽象類體現(xiàn)了數(shù)據(jù)抽象的思想(不然呢),是實現(xiàn)多態(tài)的一種機制。抽象類定義了一組抽象的方法,至于這組抽象方法的具體表現(xiàn)形式由子類來繼承實現(xiàn)。
抽象類就是用來繼承的,否則它就沒有存在的任何意義。舉個例子,我們來定義一個抽象的作者類。
abstract class Author {abstract void write ();public void sleep () {System.out.println("吃飯睡覺打豆豆");}}
作為一名作者,本職工作就是搞寫作的,其他時間就吃飯睡覺打豆豆;但至于能寫出什么樣的作品,就要看是哪一個作者了。比如說,沉默王二能寫出的作品一定是幽默風(fēng)趣的。
public class Wanger extends Author {@Overridevoid write() {System.out.println("沉默王二的作品《Web 全棧開發(fā)進階之路》,讀起來輕松愜意");}}
注意到了沒?抽象類是可以有自己的方法的,但繼承它的子類可以忽視。
接口
接口是一種比抽象類更加抽象的“類”,畢竟是用關(guān)鍵字 interface 聲明的,不是用 class。
接口只是一種形式,就好像一紙契約,自身不能做任何事情。但只要某個類實現(xiàn)了這個接口,就必須按照這紙契約來辦事:接口里提到的方法必須全部實現(xiàn),少一個都不行(抽象類的子類可以忽視非抽象方法)。舉個例子,我們來定義一個北航出版合同的接口。
interface ContractBeihang {void scriptBeihang();}
一旦作者簽訂了合同,那么就必須定期完成一定量的書稿。
public class Wanger extends Author implements ContractBeihang {@Overridevoid write() {System.out.println("作品《Web 全棧開發(fā)進階之路》,讀起來輕松愜意的技術(shù)書");}@Overridepublic void scriptBeihang() {System.out.println("一年內(nèi)完成書稿啊,不然要交違約金的哦。");}}
接口是抽象類的補充,Java 為了保證數(shù)據(jù)的安全性不允許多重繼承,也就是說一個類同時只允許繼承一個父類(為什么呢?請搜索關(guān)鍵字“菱形問題”)。
但是接口不同,一個類可以同時實現(xiàn)多個接口,這些接口之間可以沒有多大的關(guān)系(彌補了抽象類不能多重繼承的缺陷)。比如說,沉默王二不僅簽了北航出版社的合同,還和 51CTO 簽了付費課程的合同。
public class Wanger extends Author implements ContractBeihang, Contract51 {@Overridevoid write() {System.out.println("作品《Web 全棧開發(fā)進階之路》,讀起來輕松愜意的技術(shù)書");}@Overridepublic void scriptBeihang() {System.out.println("一年內(nèi)完成書稿啊,不然要交違約金的哦。");}@Overridepublic void script51() {System.out.println("王老師,先把 Java 云盤的大綱整理出來。");}}
03、接口和抽象類之間的差別
通過上面舉的例子,是不是對接口和抽象類有比較清晰的認知了?如果還沒有,來來來,我們再來比較一下接口和抽象類之間的差別。
究竟什么時候使用接口,什么時候使用抽象類呢?
1、抽象類表示了一種“is-a”的關(guān)系,而接口表示的是“like-a”的關(guān)系。也就是說,如果 B 類是 A(沉默王二是一個作者),則 A 應(yīng)該用抽象類。如果 B 類只是和 A 有某種關(guān)系,則 A 應(yīng)該用接口。
2、 如果要擁有自己的成員變量和非抽象方法,則用抽象類。接口只能存在靜態(tài)的不可變的成員變量(不過一般都不在接口中定義成員變量)。
3、為接口添加任何方法(抽象的),相應(yīng)的所有實現(xiàn)了這個接口的類,也必須實現(xiàn)新增的方法,否則會出現(xiàn)編譯錯誤。對于抽象類,如果添加了非抽象方法,其子類卻可以坐享其成,完全不必擔(dān)心編譯會出問題。
4、抽象類和接口有很大的相似性,請謹(jǐn)慎判斷。Java 從1.8版本開始,嘗試向接口中引入了默認方法和靜態(tài)方法,以此來減少抽象類和接口之間的差異。換句話說,兩者之間越來越難區(qū)分了。
04、接口的實際應(yīng)用
在實際的開發(fā)應(yīng)用當(dāng)中,抽象類我用得不多(這可真是大實話);接口我倒是用得蠻多的,就像下面這樣子:
public interface CityMapper {@Select("select * from city")List
@Insert、@Update、@Delete、@Select 被稱為 Mybatis 的注射器注解。
是不是突然感覺有點懵?之前還在談接口和抽象類,怎么一下子跳躍到 Mybatis 上面了呢?還有什么映射器注解?
嗯,這就對了。所有的理論知識都要應(yīng)用于實踐,否則也就沒有了存在價值。在我的實踐應(yīng)用當(dāng)中,接口用得多的就是 Mybatis 的 Mapper 接口。
MyBatis 是一款優(yōu)秀的持久層框架,它支持定制化 SQL、存儲過程以及高級映射。MyBatis 避免了幾乎所有的 JDBC 代碼和手動設(shè)置參數(shù)以及獲取結(jié)果集。MyBatis 可以使用簡單的 XML 或注解(就是你在前面見到的增刪改查四大注解)來配置和映射原生類型、接口和 Java 的 POJO(Plain Old Java Objects,普通老式 Java 對象)為數(shù)據(jù)庫中的記錄。
當(dāng)我們配置好了 MyBatis 環(huán)境后,可以直接通過以下語句來調(diào)用注射器接口。
@Servicepublic class CityService {@Autowiredprivate CityMapper cityMapper;public void init() {List
在注射器接口中,也只會存在那些與數(shù)據(jù)庫查詢相關(guān)的抽象方法,就像你看到的 List
05、總結(jié)
這篇文章的目的是幫助更多的讀者了解和掌握抽象類、接口的特點,以及不同的使用場景。