Java 8中default方法能做什么?不能做什么?
什么是default方法?
Java 8 發(fā)布以后,可以給接口添加新方法,但是,接口仍然可以和它的實現(xiàn)類保持兼容。這非常重要,因為你開發(fā)的類庫可能正在被多個開發(fā)者廣泛的使用著。而Java 8之前,在類庫中發(fā)布了一個接口以后,如果在接口中添加一個新方法,那些實現(xiàn)了這個接口的應(yīng)用使用新版本的接口就會有崩潰的危險。
有了Java 8,是不是就沒有這種危險了?答案是否定的。
給接口添加 default 方法可能會讓某些實現(xiàn)類不可用。
首先,讓我們看下 default 方法的細(xì)節(jié)。
在Java 8中,接口中的方法可以被實現(xiàn)(Java8中的 static 的方法也可以在接口中實現(xiàn),但這是另一個話題)。接口中被實現(xiàn)的方法叫做 default 方法,用關(guān)鍵字 default 作為修飾符來標(biāo)識。當(dāng)一個類實現(xiàn)一個接口的時候,它可以實現(xiàn)已經(jīng)在接口中被實現(xiàn)過的方法,但這不是必須的。這個類會繼承 default 方法。這就是為什么當(dāng)接口發(fā)生改變的時候,實現(xiàn)類不需要做改動的原因。
多繼承的時候呢?
當(dāng)一個類實現(xiàn)了多于一個(比如兩個)接口,而這些接口又有同樣的 default 方法的時候,事情就變得很復(fù)雜了。類繼承的是哪一個 default 方法呢?哪一個也不是!在這種情況下,類要自己(直接或者是繼承樹上更上層的類)來實現(xiàn) default 方法(才可以)。
當(dāng)一個接口實現(xiàn)了 default 方法,另一個接口把 default 方法聲明成了 abstract 的時候,同樣如此。Java 8試圖避免不明確的東西,保持嚴(yán)謹(jǐn)。如果一個方法在多個接口中都有聲明,那么,任何一個 default 實現(xiàn)都不會被繼承,你將會得到一個編譯時錯誤。
但是,如果你已經(jīng)把你的類編譯過了,那就不會出現(xiàn)編譯時錯誤了。在這一點上,Java 8是不一致的。它有它自己的原因,有于各種原因,在這里我不想詳細(xì)的說明或者是深入的討論(因為:版本已經(jīng)發(fā)布了,討論時間太長,這個平臺從來沒有這樣的討論)。
- 假如你有兩個接口,一個實現(xiàn)類。
- 其中一個接口實現(xiàn)了一個 default 方法 m() 。
- 把接口和實現(xiàn)類一塊編譯。
- 修改那個沒有包含 m() 方法的接口,聲明 m() 方法為 abstract 。
- 單獨重新編譯修改過的接口。
- 運行實現(xiàn)類。

上面的情況下類可以正常運行。但是,不能用修改過的接口重新編譯,但是用老的接口編譯仍然可以運行。接下來
- 修改那個含有 abstract 方法 m() 的接口,創(chuàng)建一個 default 實現(xiàn)。
- 編譯修改后的接口
- 運行類:失敗。
當(dāng)兩個接口給同一個方法都提供了default實現(xiàn)的時候,這個方法是無法被調(diào)用的,除非實現(xiàn)類也實現(xiàn)了這個default方法(要么是直接實現(xiàn),要么是繼承樹上更上層的類做實現(xiàn))。

但是,這個類是兼容的。它可以在使用新接口的情況下被載入,甚至可以執(zhí)行,只要它沒有調(diào)用在兩個接口中都有 default 實現(xiàn)的方法。
實例代碼

為了演示上面的例子,我給 C.java 創(chuàng)建了一個測試目錄,它下面還有3個子目錄,用于存放 I1.java 和 I2.java 。測試目錄下包含了類C的源碼 C.java 。base目錄包含了可以編譯和運行的那個版本的接口。I1包含了有 default 實現(xiàn)的 m() 方法, I2 不包含任何方法。
實現(xiàn)類包含了 main 方法,所以我們可以在測試中執(zhí)行它。它會檢查是否存在命令行參數(shù),這樣,我們就可以很方便的執(zhí)行調(diào)用 m() 和不調(diào)用 m() 的測試。
public class C implements I1, I2 {
public static void main(String[] args) {
C c = new C();
if(args.length == 0 ){
c.m();
}
}
}
public interface I1 {
default void m(){
System.out.println("hello interface 1");
}
}
public interface I2 {
}
使用下面的命令行來編譯運行:
javac -cp .:base C.java java -cp .:base C hello interface 1
compatible 目錄包含了有 abstract 方法 m() 的 I2 接口,和未修改的 I1 接口。
public interface I2 {
void m();
}
這個不能用來編譯類C:
javac -cp .:compatible C.java
C.java:1: error: C is not abstract and does not override abstract method m() in I2
public class C implements I1, I2 {
^
1 error
錯誤信息非常精確。因為我們有前一次編譯獲得的 C.class ,如果我們編譯 compatible 目錄下的接口,我們?nèi)匀粫玫侥苓\行實現(xiàn)類的兩個接口:
javac compatible/I*.java java -cp .:compatible C hello interface 1
第三個叫做 wrong 的目錄,包含的 I2 接口也定義了 m() 方法:
public interface I2 {
default void m(){
System.out.println("hello interface 2");
}
}
我們應(yīng)該不厭其煩的編譯它。盡管m()方法被定義了兩次,但是,實現(xiàn)類仍然可以運行,只要它沒有調(diào)用那個定義了多次的方法,但是,只要我們調(diào)用m()方法,立即就會失敗。這是我們使用的命令行參數(shù):
javac wrong/*.java java -cp .:wrong C Exception in thread "main" java.lang.IncompatibleClassChangeError: Conflicting default methods: I1.m I2.m at C.m(C.java) at C.main(C.java:5) java -cp .:wrong C x
結(jié)論
當(dāng)你把給接口添加了 default 實現(xiàn)的類庫移植到Java 8環(huán)境下的時候,一般不會有問題。至少Java8類庫開發(fā)者給集合類添加default方法的時候就是這么想的。使用你類庫的應(yīng)用程序仍然依賴沒有 default 方法的Java7的類庫。當(dāng)使用和修改多個不同的類庫的時候,有很小的幾率會發(fā)生沖突。如何才能避免呢?
像以前那樣設(shè)計你的類庫??赡芤蕾?default 方法的時候不要掉以輕心。萬不得已不要使用。明智的選擇方法名,避免和其它接口產(chǎn)生沖突。我們將會學(xué)習(xí)到Java編程中如何使用這個特性做開發(fā)。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
Spring Boot 整合 ShedLock 處理定時任務(wù)重復(fù)執(zhí)行的問題小結(jié)
ShedLock是解決分布式系統(tǒng)中定時任務(wù)重復(fù)執(zhí)行問題的Java庫,通過在數(shù)據(jù)庫中加鎖,確保只有一個節(jié)點在指定時間執(zhí)行任務(wù),它與SpringScheduler、Quartz等框架結(jié)合使用,本文介紹Spring Boot 整合 ShedLock 處理定時任務(wù)重復(fù)執(zhí)行的問題,感興趣的朋友一起看看吧2025-02-02
Spring?Cloud中Sentinel的兩種限流模式介紹
如何使用Sentinel做流量控制呢?這篇文章就來為大家詳細(xì)介紹了Spring?Cloud中Sentinel的兩種限流模式,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-05-05
詳解mall整合SpringBoot+MyBatis搭建基本骨架
這篇文章主要介紹了詳解mall整合SpringBoot+MyBatis搭建基本骨架,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
MyBatis-Plus 動態(tài)表名SQL解析器的實現(xiàn)
這篇文章主要介紹了MyBatis-Plus 動態(tài)表名SQL解析器的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Springboot+hibernate實現(xiàn)簡單的增刪改查示例
今天小編就為大家分享一篇Springboot+hibernate實現(xiàn)簡單的增刪改查示例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-08-08

