Java 8的default方法能作什麼?不能作什麼?

什麼是default方法?

Java 8 發佈之後,能夠給接口添加新方法,可是,接口仍然能夠和它的實現類保持兼容。這很是重要,由於你開發的類庫可能正在被多個開發者普遍的使用着。而Java 8以前,在類庫中發佈了一個接口之後,若是在接口中添加一個新方法,那些實現了這個接口的應用使用新版本的接口就會有崩潰的危險。html

有了Java 8,是否是就沒有這種危險了?答案是否認的。java

給接口添加default方法可能會讓某些實現類不可用。編程

首先,讓咱們看下default方法的細節。bash

在Java 8中,接口中的方法能夠被實現(Java8中的static的方法也能夠在接口中實現,但這是另外一個話題)。接口中被實現的方法叫作default方法,用關鍵字default做爲修飾符來標識。當一個類實現一個接口的時候,它能夠實現已經在接口中被實現過的方法,但這不是必須的。這個類會繼承default方法。這就是爲何當接口發生改變的時候,實現類不須要作改動的緣由。ide

多繼承的時候呢?

當一個類實現了多於一個(好比兩個)接口,而這些接口又有一樣的default方法的時候,事情就變得很複雜了。類繼承的是哪個default方法呢?哪個也不是!在這種狀況下,類要本身(直接或者是繼承樹上更上層的類)來實現default方法(才能夠)。學習

當一個接口實現了default方法,另外一個接口把default方法聲明成了abstract的時候,一樣如此。Java 8試圖避免不明確的東西,保持嚴謹。若是一個方法在多個接口中都有聲明,那麼,任何一個default實現都不會被繼承,你將會獲得一個編譯時錯誤。測試

可是,若是你已經把你的類編譯過了,那就不會出現編譯時錯誤了。在這一點上,Java 8是不一致的。它有它本身的緣由,有於各類緣由,在這裏我不想詳細的說明或者是深刻的討論(由於:版本已經發布了,討論時間太長,這個平臺歷來沒有這樣的討論)。spa

  • 假如你有兩個接口,一個實現類。
  • 其中一個接口實現了一個default方法m()
  • 把接口和實現類一塊編譯。
  • 修改那個沒有包含m()方法的接口,聲明m()方法爲abstract
  • 單獨從新編譯修改過的接口。
  • 運行實現類。

上面的狀況下類能夠正常運行。可是,不能用修改過的接口從新編譯,可是用老的接口編譯仍然能夠運行。接下來命令行

  • 修改那個含有abstract方法m()的接口,建立一個default實現。
  • 編譯修改後的接口
  • 運行類:失敗。

當兩個接口給同一個方法都提供了default實現的時候,這個方法是沒法被調用的,除非實現類也實現了這個default方法(要麼是直接實現,要麼是繼承樹上更上層的類作實現)。翻譯

可是,這個類是兼容的。它能夠在使用新接口的狀況下被載入,甚至能夠執行,只要它沒有調用在兩個接口中都有default實現的方法。

實例代碼

爲了演示上面的例子,我給C.java建立了一個測試目錄,它下面還有3個子目錄,用於存放I1.javaI2.java。測試目錄下包含了類C的源碼C.java。base目錄包含了能夠編譯和運行的那個版本的接口。I1包含了有default實現的m()方法,I2不包含任何方法。

實現類包含了main方法,因此咱們能夠在測試中執行它。它會檢查是否存在命令行參數,這樣,咱們就能夠很方便的執行調用m()和不調用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目錄下的接口,咱們仍然會獲得能運行實現類的兩個接口:

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");
    }
}
複製代碼

咱們應該不厭其煩的編譯它。儘管m()方法被定義了兩次,可是,實現類仍然能夠運行,只要它沒有調用那個定義了屢次的方法,可是,只要咱們調用m()方法,當即就會失敗。這是咱們使用的命令行參數:

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 
複製代碼

結論

當你把給接口添加了default實現的類庫移植到Java 8環境下的時候,通常不會有問題。至少Java8類庫開發者給集合類添加default方法的時候就是這麼想的。使用你類庫的應用程序仍然依賴沒有default方法的Java7的類庫。當使用和修改多個不一樣的類庫的時候,有很小的概率會發生衝突。如何才能避免呢?

像之前那樣設計你的類庫。可能依賴default方法的時候不要掉以輕心。萬不得已不要使用。明智的選擇方法名,避免和其它接口產生衝突。咱們將會學習到Java編程中如何使用這個特性作開發。


歡迎關注知乎專欄《跟上Java8》,分享優秀的Java8中文指南、教程,同時歡迎投稿高質量的文章。

原文連接: javacodegeeks
翻譯: ImportNew.com - miracle1919
譯文連接: www.importnew.com/10764.html
相關文章
相關標籤/搜索