Java8中的默認方法

做者:湯圓java

我的博客:javalover.ccide

前言

你們好啊,我是湯圓,今天給你們帶來的是《Java8中的默認方法》,但願對你們有幫助,謝謝url

文章純屬原創,我的總結不免有差錯,若是有,麻煩在評論區回覆或後臺私信,謝啦.net

簡介

在Java以前,咱們接觸到的接口,都是隻定義方法,不實現方法設計

(你看下面這幾我的,像不像接口)code

指手畫腳

可是到了Java8就不同了,由於在接口中新增了默認方法繼承

這樣的話,有些活,就能夠交給接口本身去作了,而不用實現類去作(Java你這是在收買人心啊)接口

咱們下面以問答的形式來介紹默認方法的相關知識點(聽說問答模式可讓人更好地記憶?)get

目錄

  • 什麼是默認方法?編譯器

  • 爲啥要提供默認實現?

  • 那我若是不提供呢?

  • 這個功能主要是針對誰?

  • 實現了默認方法的接口和抽象類有區別嗎?

  • 是否是能夠說實現了多重繼承?

正文

什麼是默認方法

默認方法是接口中用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()

這就是默認方法的好處:對實現類來講是無痛升級的

若是不提供呢?

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

  1. 實現類升級:
    • 實現類老老實實地按照接口升級後的方法,進行同步修改實現,可是工做量大
  2. 實現類不升級:
    • 實現類不升級也是能夠的,只要不引入接口類的新版本就能夠了,那麼這個時候系統仍是能夠運行的,這沒啥問題。可是誰能保證一生都不更新系統呢?若是更新系統時,接口類庫升級到新版本,那麼編譯仍是通不過

主要針對誰?

接口的默認方法主要是針對類庫設計者

實現了默認方法的接口和抽象類有區別嗎

區別沒有以前那麼多,但仍是有的:

  1. 抽象類單繼承,接口類多實現
  2. 抽象類中的屬性定義時不須要初始化,接口類的屬性定義時要初始化(默認修飾符爲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修飾且包含方法內容的方法

  • 爲何要提供默認方法:向後兼容,使系統平滑過渡;主要針對類庫設計者

  • 多重繼承帶來的問題:二義性,也叫菱形問題;解決辦法就是子類儘可能覆寫默認方法並顯式聲明調用哪一個方法(實際上這個問題不多出現,由於它屬於編譯錯誤,寫代碼時隨時能夠發現)

後記

最後,感謝你們的觀看,謝謝

相關文章
相關標籤/搜索