作者:湯圓
個人博客:javalover.cc
前言
大家好啊,我是湯圓,今天給大家帶來的是《Java8中的默認方法》,希望對大家有幫助,謝謝
文章純屬原創,個人總結難免有差錯,如果有,麻煩在評論區回復或后台私信,謝啦
簡介
在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修飾且包含方法內容的方法
-
為什么要提供默認方法:向后兼容,使系統平滑過渡;主要針對類庫設計者
-
多重繼承帶來的問題:二義性,也叫菱形問題;解決辦法就是子類盡量覆寫默認方法並顯式聲明調用哪個方法(實際上這個問題很少出現,因為它屬於編譯錯誤,寫代碼時隨時可以發現)
后記
最后,感謝大家的觀看,謝謝