首頁 > 軟體

Java8新特性之預設方法詳解

2022-02-16 13:00:38

簡介

在Java之前,我們接觸到的介面,都是隻定義方法,不實現方法

(你看下面這幾個人,像不像介面)

但是到了Java8就不一樣了,因為在介面中新增了預設方法

這樣的話,有些活,就可以交給介面自己去做了,而不用實現類去做(Java你這是在收買人心啊)

我們下面以問答的形式來介紹預設方法的相關知識點(據說問答模式可以讓人更好地記憶?)

正文

什麼是預設方法

預設方法是介面中用default修飾的方法,其中包含方法內容

比如下面這個:

public interface InterfaceDemo {
   // 普通方法,只定義,不實現
   void oldFun();
   // 預設方法,又定義,又實現
   default void newFun(){
       System.out.println("newFun");
  }
}

為啥要提供預設方法呢?

為了向後相容(這也是導致Java變得臃腫的原因之一)。

因為升級系統時,難免會有一些新功能需要加入,此時如果介面類新增了方法,那麼實現類就必須同步修改實現;

這樣工作量還是很大的,而且很容易出錯。

所以Java8開始,推出了介面的預設方法這個功能,使得介面升級變得更加平滑

比如下面的程式碼:InterfaceDemo就是上面那個介面

public class UserDemo implements InterfaceDemo{
   @Override
   public void oldFun() {
       System.out.println("oldFun");
  }

   public static void main(String[] args) {
       UserDemo demo = new UserDemo();
       /**
        * InterfaceDemo升級後,新增了newFun方法
        * 但是由於newFun是預設方法,有提供實現內容
        * 所以這裡的子類 UserDemo就可以直接使用
         */
       demo.newFun();
  }
}

我們可以看到,UserDemo沒有實現新的方法newFun(),但是也可以編譯執行,並直接呼叫newFun()

這就是預設方法的好處:對實現類來說是無痛升級的

如果不提供呢?

不提供的話,介面類升級時,系統有兩個選擇

  • 實現類升級:

    • 實現類老老實實地按照介面升級後的方法,進行同步修改實現,但是工作量大

  • 實現類不升級:

    • 實現類不升級也是可以的,只要不引入介面類的新版本就可以了,那麼這個時候系統還是可以執行的,這沒啥問題。但是誰能保證一輩子都不更新系統呢?如果更新系統時,介面類庫升級到新版本,那麼編譯還是通不過

主要針對誰?

介面的預設方法主要是針對類庫設計者

實現了預設方法的介面和抽象類有區別嗎

區別沒有之前那麼多,但還是有的:

  • 抽象類單繼承,介面類多實現

  • 抽象類中的屬性定義時不需要初始化,介面類的屬性定義時要初始化(預設修飾符為public static final)

是不是可以說Java現在也實現了多重繼承?

可以這麼說。

但是現在面臨的一個新問題,就是多重繼承帶來的二義性問題,有點類似之前介紹的致命方塊(也叫菱形問題)

如下面的UML圖所示

比如上面這種,你無法知道A會呼叫哪個介面的fun方法

所以編譯器會報錯:

com.jalon.java8.defaultmethod.A inherits unrelated defaults for fun() from types com.jalon.java8.defaultmethod.B and com.jalon.java8.defaultmethod.C

解決辦法:

  • 先覆寫fun方法

  • 再顯示宣告呼叫哪個介面的fun方法

程式碼如下:

public class A implements B,C{
   @Override
   public void fun(){
       // 顯示呼叫B的預設方法
       B.super.fun();
  }

   public static void main(String[] args) {
       A a = new A();
       // 這裡會列印B的fun
       a.fun();
  }
}
interface D{
   default void fun(){
       System.out.println("D");
  }
}
interface B extends D{
   @Override
   default void fun(){
       System.out.println("B");
  }
}
interface C extends D{
   @Override
   default void fun(){
       System.out.println("C");
  }
}

總結

  • 什麼是預設方法:介面中用default修飾且包含方法內容的方法

  • 為什麼要提供預設方法:向後相容,使系統平滑過渡;主要針對類庫設計者

  • 多重繼承帶來的問題:二義性,也叫菱形問題;解決辦法就是子類儘量覆寫預設方法並顯式宣告呼叫哪個方法(實際上這個問題很少出現,因為它屬於編譯錯誤,寫程式碼時隨時可以發現)

到此這篇關於Java8新特性之預設方法的文章就介紹到這了,更多相關Java8預設方法內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com