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