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