溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

怎么在Java8中歲接口的特性進(jìn)行測試

發(fā)布時間:2021-03-09 16:16:38 來源:億速云 閱讀:106 作者:Leah 欄目:編程語言

怎么在Java8中歲接口的特性進(jìn)行測試?相信很多沒有經(jīng)驗的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。

默認(rèn)方法

方法前加default關(guān)鍵字就可以提供默認(rèn)實現(xiàn),類實現(xiàn)接口時,可以繼承接口的默認(rèn)方法,也可以覆蓋默認(rèn)方法。

interface People {
  default void eat(String name) {
    System.out.println(name + " is eating.");
  }
}

抽象類也可以提供方法的默認(rèn)實現(xiàn),一個類可以同時繼承一個抽象類和多個接口,如果抽象類和接口中存在相同的方法時會怎么樣呢?

一個抽象類和多個接口

如果類繼承了抽象類,采取類優(yōu)先的原則,優(yōu)先繼承抽象類的方法。我們在編寫類的時候會發(fā)現(xiàn),必須先extends類再implements接口,否則工具會提示extends的錯誤。不論抽象類的方法有沒有提供默認(rèn)實現(xiàn),類都會繼承抽象類的方法,就算接口的方法提供了默認(rèn)實現(xiàn),也完全不關(guān)心。

接口A,B:

interface A {
  default void eat() {
    System.out.println("A");
  }
}
interface B {
  default void eat() {
    System.out.println("B");
  }
}

抽象類C:

abstract class C {
  public void eat() {
    System.out.println("C");
  }
}

測試類Test:

public class Test extends C implements A, B {
 public static void main(String[] args) {
    Test test = new Java8Test();
    test.eat();
  }
}

輸出結(jié)果:

C

程序輸出C說明Test類繼承了抽象類C的eat方法的實現(xiàn)。

當(dāng)抽象類C不提供eat方法的默認(rèn)實現(xiàn)并且Test類不提供eat方法的重寫時,編譯器是不會放過Test類的,明明繼承了C類,卻不提供抽象方法的實現(xiàn)。編譯器不會關(guān)心Test是否還實現(xiàn)了其他接口并且接口中有相同方法,一碼歸一碼。

那么如果是抽象類提供了默認(rèn)實現(xiàn),接口沒有提供呢?

我們讓抽象類C實現(xiàn)eat方法,接口A只提供一個接口,類Test不重寫eat方法。

接口A,B:

interface A {
  void eat();
}
interface B {
  default void eat() {
    System.out.println("B");
  }
}

抽象類C:

abstract class C {
  public void eat() {
    System.out.println("C");
  }
}

測試類Test:

public class Test extends C implements A, B {
 public static void main(String[] args) {
    Test test = new Test();
    test.eat();
  }
}

運行main方法,控制臺打印出抽象類中eat方法的輸出結(jié)果。

其實我們可以這么理解,編譯器總要一步一步的編譯,首先編譯到extends,檢查所有的抽象方法是否被重寫,有沒被重寫的方法就拋出編譯錯誤,否則就繼續(xù)編譯。當(dāng)編譯到implements時,類Test已經(jīng)是擁有抽象類C所有方法的實現(xiàn)了,即已經(jīng)提供了接口A和B中相同方法的重寫。

以上總結(jié)就是抽象類和接口的方法沖突時以抽象類為準(zhǔn)就是了。如果抽象類不在了,兩個接口方法沖突了,該聽誰的呢?

多個接口

接口A提供方法默認(rèn)實現(xiàn),接口B只提供方法。

接口A、B:

interface A {
   default void eat() {
     System.out.println("A");
   }
}
interface B {
  default void eat() {
     System.out.println("B");
   }
}

測試類Test:

public class Test implements A, B {
  public static void main(String[] args) {
    Test test = new Test();
    test.eat();
  }
}

此時編譯器提示,類Test從類型A和B中繼承了eat()的不相關(guān)默認(rèn)值。也就是說,當(dāng)兩個接口沖突時,編譯器無法判斷繼承A還是繼承B的方法實現(xiàn)。這時候,類Test就需要自己提供eat()方法的實現(xiàn)了。

靜態(tài)方法

接口中也可以提供靜態(tài)方法,使用static關(guān)鍵字。類不能繼承接口的靜態(tài)方法,所以也不存在覆蓋靜態(tài)方法。訪問靜態(tài)方法時通過接口訪問,即People.eat(“name”)。

interface People {
  static void eat(String name) {
    System.out.println(name + " is eating.");
  }
}

可以測試一下,調(diào)用繼承了People接口的類實例的eat方法,工具提示不能通過類實例訪問靜態(tài)成員?;蛘咴囍貙懸粋€eat方法并加上@Override注解,工具提示“Method does not override method from its superclass”。

看完上述內(nèi)容,你們掌握怎么在Java8中歲接口的特性進(jìn)行測試的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI