前些日有同學問我Java接口中為什么要增加一個默認方法。在Java 8 中不單單增加了默認接口方法還增加了靜態(tài)接口方法。今天就來聊聊這兩個方法。
1、默認接口方法
java中的默認接口方法是這樣定義的:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public interface NewInterface { ? void otherMethod(); default void doSomething() { System.out.println( " do something " ); } } ? public class NewInterfaceImpl implements NewInterface { @Override public void otherMethod() { ? } } |
在接口中使用關鍵字default
聲明并提供具體實現(xiàn),而且該方法不需要添加public
關鍵字就可以公開調(diào)用,甚至你可以在其實現(xiàn)類中覆寫( @Override
)。
通常基于抽象的概念設計中,抽象往往有多個實現(xiàn),如果沒有多個實現(xiàn)也不太必要去抽象。而且一個接口的實現(xiàn)必須實現(xiàn)該接口所有的抽象方法,這并不是一個可選擇的行為,如果我們在一個接口中增加了新的方法就會影響到其所有的實現(xiàn)類,很可能有些實現(xiàn)類根本不需要這個功能,但是又不得不去實現(xiàn)該功能。
接口默認方法是解決這個問題的有效方式。它允許在接口中直接添加一個功能實現(xiàn),并且所有的實現(xiàn)類都可以使用這個功能實現(xiàn)而無需修改這些實現(xiàn)類。通過這種方式,可以巧妙地保留向后兼容性。另外這也是擴展函數(shù)式接口的唯一方法,如果沒有默認方法,一旦函數(shù)式接口需要擴展就會引發(fā)“災難性”的連鎖反應。
這樣說來默認接口方法確實是一個非常好的特性,但是還是有一些需要我們注意的問題。由于 Java
允許類實現(xiàn)多個接口,當一個類實現(xiàn)多個定義相同默認方法的接口時會發(fā)生什么情況? 我們再定義一個帶默認接口方法的接口,并且方法的名稱也是doSomething
:
1
2
3
4
5
6
7
|
public interface AnotherInterface { void anotherMethod(); ? default void doSomething() { System.out.println( " do another something " ); } } |
這時候如果一個類同時實現(xiàn)了NewInterface
和AnotherInterface
方法簽名引起的沖突:
因為方法簽名相同,無法判斷使用哪個方法,連編譯都過不去。為了解決這個分歧,我們必須明確地為這些方法提供一個實現(xiàn)。
2、靜態(tài)接口方法
從 Java 8 開始,我們還可以在接口中定義靜態(tài)方法。由于靜態(tài)接口方法不屬于特定對象,因此它們不是接口實現(xiàn)類的一部分,你不能通過實現(xiàn)類去調(diào)用靜態(tài)接口方法,只能通過接口去調(diào)用。利用這個特性我們可以實現(xiàn)一些固定范式的功能,并且該功能不會因為接口的多態(tài)而改變,避免出現(xiàn)不良后果。
1
2
3
4
5
6
7
8
9
10
11
|
@FunctionalInterface public interface Customizer<T> { void customize(T t); static <T> Customizer<T> withDefaults() { return (t) -> { }; } ? } |
這是Spring Security
自定義配置的關鍵接口,如果你使用默認配置你可以直接通過靜態(tài)方法實現(xiàn),如果你想自定義就通過實現(xiàn)抽象接口,這樣統(tǒng)一的范式就能夠穩(wěn)定下來了,一般作為一種功能的輔助手段,而且這種手段不能被改寫。
3、兩者的場景差異
- 默認接口方法 提供默認的功能實現(xiàn),你不喜歡可以改。
- 靜態(tài)接口方法 同樣提供一個默認的功能實現(xiàn),對不起愛用不用。
到此這篇關于Java8 接口默認方法和靜態(tài)方法的文章就介紹到這了,更多相關Java8 接口默認方法和靜態(tài)方法內(nèi)容請搜索服務器之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持服務器之家!
原文鏈接:https://juejin.cn/post/7022900705983725582