Java8中接口的新特性測試
在Java SE 8之前,接口中是不能提供方法實(shí)現(xiàn)的,但是JDK8提供了接口的默認(rèn)方法和靜態(tài)方法的支持。
默認(rèn)方法
方法前加default關(guān)鍵字就可以提供默認(rèn)實(shí)現(xiàn),類實(shí)現(xiàn)接口時(shí),可以繼承接口的默認(rèn)方法,也可以覆蓋默認(rèn)方法。
interface People { default void eat(String name) { System.out.println(name + " is eating."); } }
抽象類也可以提供方法的默認(rèn)實(shí)現(xiàn),一個(gè)類可以同時(shí)繼承一個(gè)抽象類和多個(gè)接口,如果抽象類和接口中存在相同的方法時(shí)會(huì)怎么樣呢?
一個(gè)抽象類和多個(gè)接口
如果類繼承了抽象類,采取類優(yōu)先的原則,優(yōu)先繼承抽象類的方法。我們在編寫類的時(shí)候會(huì)發(fā)現(xiàn),必須先extends類再implements接口,否則工具會(huì)提示extends的錯(cuò)誤。不論抽象類的方法有沒有提供默認(rèn)實(shí)現(xiàn),類都會(huì)繼承抽象類的方法,就算接口的方法提供了默認(rèn)實(shí)現(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方法的實(shí)現(xiàn)。
當(dāng)抽象類C不提供eat方法的默認(rèn)實(shí)現(xiàn)并且Test類不提供eat方法的重寫時(shí),編譯器是不會(huì)放過Test類的,明明繼承了C類,卻不提供抽象方法的實(shí)現(xiàn)。編譯器不會(huì)關(guān)心Test是否還實(shí)現(xiàn)了其他接口并且接口中有相同方法,一碼歸一碼。
那么如果是抽象類提供了默認(rèn)實(shí)現(xiàn),接口沒有提供呢?
我們讓抽象類C實(shí)現(xiàn)eat方法,接口A只提供一個(gè)接口,類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(); } }
運(yùn)行main方法,控制臺打印出抽象類中eat方法的輸出結(jié)果。
其實(shí)我們可以這么理解,編譯器總要一步一步的編譯,首先編譯到extends,檢查所有的抽象方法是否被重寫,有沒被重寫的方法就拋出編譯錯(cuò)誤,否則就繼續(xù)編譯。當(dāng)編譯到implements時(shí),類Test已經(jīng)是擁有抽象類C所有方法的實(shí)現(xiàn)了,即已經(jīng)提供了接口A和B中相同方法的重寫。
以上總結(jié)就是抽象類和接口的方法沖突時(shí)以抽象類為準(zhǔn)就是了。如果抽象類不在了,兩個(gè)接口方法沖突了,該聽誰的呢?
多個(gè)接口
接口A提供方法默認(rèn)實(shí)現(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(); } }
此時(shí)編譯器提示,類Test從類型A和B中繼承了eat()的不相關(guān)默認(rèn)值。也就是說,當(dāng)兩個(gè)接口沖突時(shí),編譯器無法判斷繼承A還是繼承B的方法實(shí)現(xiàn)。這時(shí)候,類Test就需要自己提供eat()方法的實(shí)現(xiàn)了。
靜態(tài)方法
接口中也可以提供靜態(tài)方法,使用static關(guān)鍵字。類不能繼承接口的靜態(tài)方法,所以也不存在覆蓋靜態(tài)方法。訪問靜態(tài)方法時(shí)通過接口訪問,即People.eat(“name”)。
interface People { static void eat(String name) { System.out.println(name + " is eating."); } }
可以測試一下,調(diào)用繼承了People接口的類實(shí)例的eat方法,工具提示不能通過類實(shí)例訪問靜態(tài)成員。或者試著重寫一個(gè)eat方法并加上@Override注解,工具提示“Method does not override method from its superclass”。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接
相關(guān)文章
在MyBatis中使用 # 和 $ 書寫占位符的區(qū)別說明
這篇文章主要介紹了在MyBatis中使用 # 和 $ 書寫占位符的區(qū)別說明,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-10-10Springboot中如何使用Redisson實(shí)現(xiàn)分布式鎖淺析
redisson是redis的java客戶端程序,國內(nèi)外很多公司都有在用,下面這篇文章主要給大家介紹了關(guān)于Springboot中如何使用Redisson實(shí)現(xiàn)分布式鎖的相關(guān)資料,需要的朋友可以參考下2021-10-10解讀maven項(xiàng)目啟動(dòng)tomcat不報(bào)錯(cuò)但是啟動(dòng)不起來,tomcat啟動(dòng)到警告log4j就停止了
這篇文章主要介紹了maven項(xiàng)目啟動(dòng)tomcat不報(bào)錯(cuò)但是啟動(dòng)不起來,tomcat啟動(dòng)到警告log4j就停止了問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07Java concurrency集合之ConcurrentSkipListSet_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
這篇文章主要為大家詳細(xì)介紹了Java concurrency集合之ConcurrentSkipListSet的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06Spring項(xiàng)目里將SQL語句寫在.sql文件中的方法
這篇文章主要介紹了Spring項(xiàng)目里如何將SQL語句寫在.sql文件中的方法,文中給出了詳細(xì)的介紹和示例代碼,相信對大家的學(xué)習(xí)或者工作具有一定的參考借鑒價(jià)值,有需要的朋友們下面來一起看看吧。2017-01-01SpringCloud Zuul實(shí)現(xiàn)動(dòng)態(tài)路由
這篇文章主要介紹了SpringCloud Zuul實(shí)現(xiàn)動(dòng)態(tài)路由,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-01-01