Default Methods實(shí)例解析
Oracle官網(wǎng)是這樣介紹默認(rèn)方法的,使用默認(rèn)方法,可以達(dá)到往接口里面增加新的功能,而且保持與老版本代碼兼容,也就是原來(lái)的實(shí)現(xiàn)類可以不需要被動(dòng)修改。所以,默認(rèn)方法位置是在接口里面;默認(rèn)方法具有實(shí)現(xiàn),不會(huì)強(qiáng)制具體類來(lái)現(xiàn)。Java 8要充分利用Lambda,需要增強(qiáng)大量的類庫(kù),但是又希望做到兼容性,只能用默認(rèn)方法這個(gè)大招。
默認(rèn)方法
默認(rèn)方法與普通的接口方法相比,最前面增加default關(guān)鍵字,參數(shù)列表之后接大括號(hào),實(shí)現(xiàn)該方法,再后面沒(méi)有分號(hào)。
如果翻看jdk源碼的interface,會(huì)發(fā)現(xiàn)多了default關(guān)鍵詞。
簡(jiǎn)單介紹下。
default使我們能夠在不中斷實(shí)現(xiàn)該接口的類的情況下向接口添加新的功能。 讓我們來(lái)看看下面的例子。
public class MyClass implements InterfaceA {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
// TODO code application logic here
}
@Override
public void saySomething() {
System.out.println("Hello World");
}
}
interface InterfaceA {
public void saySomething();
}
上面的代碼顯示了類MyClass實(shí)現(xiàn)InterfaceA的方法saySomething()。 現(xiàn)在我們給InterfaceA添加一個(gè)名為sayHi()的新方法。 通過(guò)這樣做,我們向類MyClass引入一個(gè)問(wèn)題,因?yàn)樗粫?huì)編譯,直到我們提供方法sayHi()的實(shí)現(xiàn)。
這時(shí)Default就有用了。 通過(guò)在方法的訪問(wèn)修飾符之前添加關(guān)鍵字default,我們不必為類MyClass中的方法sayHi()提供實(shí)現(xiàn)。
在最嚴(yán)格的意義上,default是倒退一步,因?yàn)樗鼈冊(cè)试S你用代碼“污染”你的接口。 但它們提供了最優(yōu)雅和實(shí)用的方式來(lái)允許向后兼容性。 它使jdk更容易更新所有Collections類,并為您改裝現(xiàn)有的Lambda代碼。(其實(shí)這就是default產(chǎn)生最初原因)
public class MyClass implements InterfaceA {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
// TODO code application logic here
}
@Override
public void saySomething() {
System.out.println("Hello World");
}
}
interface InterfaceA {
public void saySomething();
default public void sayHi() {
System.out.println("Hi");
}
}
注意,我們必須提供所有default方法的實(shí)現(xiàn)。 因此,default方法為我們提供了在接口中實(shí)現(xiàn)方法的靈活性。 如果具體類不提供該方法的實(shí)現(xiàn),那么實(shí)現(xiàn)將被用作默認(rèn)值。
多接口沖突
由于java中的類可以實(shí)現(xiàn)多個(gè)接口,可能會(huì)有一個(gè)情況,其中兩個(gè)或更多的接口有一個(gè)default方法具有相同的名稱,因此導(dǎo)致沖突,因?yàn)閖ava不知道一次使用什么方法。 這將導(dǎo)致編譯錯(cuò)誤:類MyClass從類型InterfaceA和InterfaceB繼承default sayHi()
讓我們來(lái)看看下面的例子。
public class MyClass implements InterfaceA, InterfaceB {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
// TODO code application logic here
}
@Override
public void saySomething() {
System.out.println("Hello World");
}
}
interface InterfaceA {
public void saySomething();
default public void sayHi() {
System.out.println("Hi from InterfaceA");
}
}
interface InterfaceB {
default public void sayHi() {
System.out.println("Hi from InterfaceB");
}
}
為了解決這樣的情況,我們必須在類MyClass中提供sayHi()方法的實(shí)現(xiàn),因此覆蓋InterfaceA和InterfaceB中的兩個(gè)方法。
public class MyClass implements InterfaceA, InterfaceB {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
// TODO code application logic here
}
@Override
public void saySomething() {
System.out.println("Hello World");
}
@Override
public void sayHi() {
System.out.println("implemetation of sayHi() in MyClass");
}
}
interface InterfaceA {
public void saySomething();
default public void sayHi() {
System.out.println("Hi from InterfaceA");
}
}
interface InterfaceB {
default public void sayHi() {
System.out.println("Hi from InterfaceB");
}
}
如果我們想要在InterfaceA或InterfaceB中專門(mén)調(diào)用一個(gè)sayHi()方法,我們也可以這樣做:
public class MyClass implements InterfaceA, InterfaceB {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
// TODO code application logic here
}
@Override
public void saySomething() {
System.out.println("Hello World");
}
@Override
public void sayHi() {
InterfaceA.super.sayHi();
}
}
interface InterfaceA {
public void saySomething();
default public void sayHi() {
System.out.println("Hi from InterfaceA");
}
}
interface InterfaceB {
default public void sayHi() {
System.out.println("Hi from InterfaceB");
}
總結(jié)
以上就是本文關(guān)于Default Methods實(shí)例解析的全部?jī)?nèi)容,希望對(duì)大家能夠有幫助。
相關(guān)文章
Java之Scanner.nextLine()讀取回車的問(wèn)題及解決
這篇文章主要介紹了Java之Scanner.nextLine()讀取回車的問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-04-04
SpringBoot集成drools的實(shí)現(xiàn)示例
本文主要介紹了SpringBoot集成drools的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05
MYSQL批量插入數(shù)據(jù)的實(shí)現(xiàn)代碼
非常的實(shí)現(xiàn)原理,代碼較多,建議大家仔細(xì)看看。2008-10-10
IDEA中Spring Initializr沒(méi)有Java8選項(xiàng)的解決辦法
在使用IDEA中的Spring Initializr創(chuàng)建新項(xiàng)目時(shí),Java 版本近可選擇Java17,21 ,不能選擇Java8;SpringBoot 版本也只有 3.x,所以本文給大家介紹了IDEA中Spring Initializr沒(méi)有Java8選項(xiàng)的解決辦法,需要的朋友可以參考下2024-06-06
java后端返回?cái)?shù)據(jù)給前端時(shí)去除值為空或NULL的屬性、忽略某些屬性代碼示例
在Java開(kāi)發(fā)中我們處理JSON數(shù)據(jù)時(shí)經(jīng)常會(huì)遇到空值(null)的情況,這篇文章主要給大家介紹了關(guān)于java后端返回?cái)?shù)據(jù)給前端時(shí)去除值為空或NULL的屬性、忽略某些屬性的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-07-07
基于Java編寫(xiě)一個(gè)神奇的代碼恢復(fù)工具
這篇文章主要為大家詳細(xì)介紹了如何基于Java編寫(xiě)一個(gè)神奇的代碼恢復(fù)工具,文中的示例代碼簡(jiǎn)潔易懂,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-04-04
了解Maven的<relativePath/>標(biāo)簽用法
這篇文章主要介紹了了解Maven的<relativePath/>標(biāo)簽用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-04-04

