Oracle官網(wǎng)是這樣介紹默認(rèn)方法的,使用默認(rèn)方法,可以達(dá)到往接口里面增加新的功能,而且保持與老版本代碼兼容,也就是原來的實(shí)現(xiàn)類可以不需要被動修改。所以,默認(rèn)方法位置是在接口里面;默認(rèn)方法具有實(shí)現(xiàn),不會強(qiáng)制具體類來現(xiàn)。Java 8要充分利用Lambda,需要增強(qiáng)大量的類庫,但是又希望做到兼容性,只能用默認(rèn)方法這個大招。
成都創(chuàng)新互聯(lián)公司基于成都重慶香港及美國等地區(qū)分布式IDC機(jī)房數(shù)據(jù)中心構(gòu)建的電信大帶寬,聯(lián)通大帶寬,移動大帶寬,多線BGP大帶寬租用,是為眾多客戶提供專業(yè)雅安服務(wù)器托管報價,主機(jī)托管價格性價比高,為金融證券行業(yè)服務(wù)器托管,ai人工智能服務(wù)器托管提供bgp線路100M獨(dú)享,G口帶寬及機(jī)柜租用的專業(yè)成都idc公司。
默認(rèn)方法
默認(rèn)方法與普通的接口方法相比,最前面增加default關(guān)鍵字,參數(shù)列表之后接大括號,實(shí)現(xiàn)該方法,再后面沒有分號。
如果翻看jdk源碼的interface,會發(fā)現(xiàn)多了default關(guān)鍵詞。
簡單介紹下。
default使我們能夠在不中斷實(shí)現(xià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(); }
上面的代碼顯示了類MyClass實(shí)現(xiàn)InterfaceA的方法saySomething()。 現(xiàn)在我們給InterfaceA添加一個名為sayHi()的新方法。 通過這樣做,我們向類MyClass引入一個問題,因?yàn)樗粫幾g,直到我們提供方法sayHi()的實(shí)現(xiàn)。
這時Default就有用了。 通過在方法的訪問修飾符之前添加關(guān)鍵字default,我們不必為類MyClass中的方法sayHi()提供實(shí)現(xiàn)。
在最嚴(yán)格的意義上,default是倒退一步,因?yàn)樗鼈冊试S你用代碼“污染”你的接口。 但它們提供了最優(yōu)雅和實(shí)用的方式來允許向后兼容性。 它使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)多個接口,可能會有一個情況,其中兩個或更多的接口有一個default方法具有相同的名稱,因此導(dǎo)致沖突,因?yàn)閖ava不知道一次使用什么方法。 這將導(dǎo)致編譯錯誤:類MyClass從類型InterfaceA和InterfaceB繼承default 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"); } } 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中的兩個方法。
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中專門調(diào)用一個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í)例解析的全部內(nèi)容,希望對大家能夠有幫助。