Java 8 接口
Java8 之前的接口定義中,方法的方法體必須是空的,接口不允許定義實(shí)例變量。并且接口實(shí)現(xiàn)類必須實(shí)現(xiàn)接口定義的所有方法。從Java 8 開始,接口可以定義默認(rèn)方法,而且所有接口實(shí)現(xiàn)類都可以改方法及實(shí)現(xiàn)。
默認(rèn)方法和抽象方法的區(qū)別是抽象方法必須要被實(shí)現(xiàn),默認(rèn)方法不是。作為替代方式,接口可以提供一個(gè)默認(rèn)的方法實(shí)現(xiàn),所有這個(gè)接口的實(shí)現(xiàn)類都會(huì)通過繼承得到這個(gè)方法(如果有需要也可以重寫這個(gè)方法)
默認(rèn)方法定義
public interface DefaultFuncInter {
??? int getInt();
??? default String getString(){
??????? return "Default String";
??? }
}
默認(rèn)方法優(yōu)勢
- 提供一種拓展接口的方法,而不破壞現(xiàn)有代碼。
假如我們有一個(gè)已經(jīng)投入使用的接口需要拓展一個(gè)新的方法,在JDK8以前,如果為一個(gè)使用的接口增加一個(gè)新方法,則我們必須在所有實(shí)現(xiàn)類中添加該方法的實(shí)現(xiàn),否則編譯會(huì)出現(xiàn)異常。如果實(shí)現(xiàn)類數(shù)量少并且我們有權(quán)限修改,可能會(huì)工作量相對(duì)較少。如果實(shí)現(xiàn)類比較多或者我們沒有權(quán)限修改實(shí)現(xiàn)類源代碼,這樣可能就比較麻煩。而默認(rèn)方法則解決了這個(gè)問題,它提供了一個(gè)實(shí)現(xiàn),當(dāng)沒有顯示提供其他實(shí)現(xiàn)時(shí)就采用這個(gè)實(shí)現(xiàn)。這樣新添加的方法將不會(huì)破壞現(xiàn)有代碼。
注意:擴(kuò)展方法不能夠重寫(也稱復(fù)寫或覆蓋) Object 中的方法,卻可以重載Object 中的方法。
- 默認(rèn)方法是可選的,子類可以根據(jù)不同的需求Override默認(rèn)實(shí)現(xiàn)。
例如,我們定義一個(gè)集合接口,其中有增、刪、改等操作。如果我們的實(shí)現(xiàn)類90%都是以數(shù)組保存數(shù)據(jù),那么我們可以定義針對(duì)這些方法給出默認(rèn)實(shí)現(xiàn),而對(duì)于其他非數(shù)組集合或者有其他類似業(yè)務(wù),可以選擇性復(fù)寫接口中默認(rèn)方法。
多重繼承的沖突說明:
由于同一個(gè)方法可以從不同接口引入,自然而然的會(huì)有沖突的現(xiàn)象,規(guī)則如下:
1)一個(gè)聲明在類里面的方法優(yōu)先于任何默認(rèn)方法
2)優(yōu)先選取最具體的實(shí)現(xiàn)
3)如果繼承多個(gè)同名的默認(rèn)方法,則必須在類中重寫默認(rèn)方法,提高優(yōu)先級(jí)。
接口靜態(tài)方法
Java8接口中可以定義靜態(tài)方法,只能通過接口名調(diào)用,不可以通過實(shí)現(xiàn)類的類名或者實(shí)現(xiàn)類的對(duì)象調(diào)用。類似類的靜態(tài)方法。
public interface DefaultFuncInter {
??? int getInt();
??? static int add(int a,int b){
??????? return a + b;
??? }
}
由于靜態(tài)方法只能通過接口來類來調(diào)用,所以不會(huì)出現(xiàn)多個(gè)接口的相同方法沖突的問題。
函數(shù)式接口
Java8開始支持Lambda表達(dá)式,一個(gè)lambda表達(dá)式都對(duì)應(yīng)一個(gè)類型,通常是接口類型。而“函數(shù)式接口”是指僅僅只包含一個(gè)抽象方法的接口,每一個(gè)該類型的lambda表達(dá)式都會(huì)被匹配到這個(gè)抽象方法。因?yàn)?默認(rèn)方法 不算抽象方法,所以你也可以給你的函數(shù)式接口添加默認(rèn)方法。
@FunctionalInterface
interface Converter<F, T> {
??? T convert(F from);
}
Converter<String, Integer> converter = (from) -> Integer.valueOf(from);
Integer converted = converter.convert("123");
System.out.println(converted);??? // 123
?
邏輯與 符號(hào)
public static <K extends Comparable<? super K>, V> Comparator<Map.Entry<K,V>> comparingByKey() {return (Comparator<Map.Entry<K, V>> & Serializable)(c1, c2) -> c1.getKey().compareTo(c2.getKey());}& 表示lambda表達(dá)式是可以轉(zhuǎn)換成的對(duì)象,可以滿足2個(gè)接口
總結(jié)
- 上一篇: 泛型的协变与逆变
- 下一篇: Java反射-继承关系