Java 8中default方法能做什么?不能做什么?
什么是default方法?
Java 8 發(fā)布以后,可以給接口添加新方法,但是,接口仍然可以和它的實(shí)現(xiàn)類(lèi)保持兼容。這非常重要,因?yàn)槟汩_(kāi)發(fā)的類(lèi)庫(kù)可能正在被多個(gè)開(kāi)發(fā)者廣泛的使用著。而Java 8之前,在類(lèi)庫(kù)中發(fā)布了一個(gè)接口以后,如果在接口中添加一個(gè)新方法,那些實(shí)現(xiàn)了這個(gè)接口的應(yīng)用使用新版本的接口就會(huì)有崩潰的危險(xiǎn)。
有了Java 8,是不是就沒(méi)有這種危險(xiǎn)了?答案是否定的。
給接口添加 default 方法可能會(huì)讓某些實(shí)現(xiàn)類(lèi)不可用。
首先,讓我們看下 default 方法的細(xì)節(jié)。
在Java 8中,接口中的方法可以被實(shí)現(xiàn)(Java8中的 static 的方法也可以在接口中實(shí)現(xiàn),但這是另一個(gè)話(huà)題)。接口中被實(shí)現(xiàn)的方法叫做 default 方法,用關(guān)鍵字 default 作為修飾符來(lái)標(biāo)識(shí)。當(dāng)一個(gè)類(lèi)實(shí)現(xiàn)一個(gè)接口的時(shí)候,它可以實(shí)現(xiàn)已經(jīng)在接口中被實(shí)現(xiàn)過(guò)的方法,但這不是必須的。這個(gè)類(lèi)會(huì)繼承 default 方法。這就是為什么當(dāng)接口發(fā)生改變的時(shí)候,實(shí)現(xiàn)類(lèi)不需要做改動(dòng)的原因。
多繼承的時(shí)候呢?
當(dāng)一個(gè)類(lèi)實(shí)現(xiàn)了多于一個(gè)(比如兩個(gè))接口,而這些接口又有同樣的 default 方法的時(shí)候,事情就變得很復(fù)雜了。類(lèi)繼承的是哪一個(gè) default 方法呢?哪一個(gè)也不是!在這種情況下,類(lèi)要自己(直接或者是繼承樹(shù)上更上層的類(lèi))來(lái)實(shí)現(xiàn) default 方法(才可以)。
當(dāng)一個(gè)接口實(shí)現(xiàn)了 default 方法,另一個(gè)接口把 default 方法聲明成了 abstract 的時(shí)候,同樣如此。Java 8試圖避免不明確的東西,保持嚴(yán)謹(jǐn)。如果一個(gè)方法在多個(gè)接口中都有聲明,那么,任何一個(gè) default 實(shí)現(xiàn)都不會(huì)被繼承,你將會(huì)得到一個(gè)編譯時(shí)錯(cuò)誤。
但是,如果你已經(jīng)把你的類(lèi)編譯過(guò)了,那就不會(huì)出現(xiàn)編譯時(shí)錯(cuò)誤了。在這一點(diǎn)上,Java 8是不一致的。它有它自己的原因,有于各種原因,在這里我不想詳細(xì)的說(shuō)明或者是深入的討論(因?yàn)椋喊姹疽呀?jīng)發(fā)布了,討論時(shí)間太長(zhǎng),這個(gè)平臺(tái)從來(lái)沒(méi)有這樣的討論)。
- 假如你有兩個(gè)接口,一個(gè)實(shí)現(xiàn)類(lèi)。
- 其中一個(gè)接口實(shí)現(xiàn)了一個(gè) default 方法 m() 。
- 把接口和實(shí)現(xiàn)類(lèi)一塊編譯。
- 修改那個(gè)沒(méi)有包含 m() 方法的接口,聲明 m() 方法為 abstract 。
- 單獨(dú)重新編譯修改過(guò)的接口。
- 運(yùn)行實(shí)現(xiàn)類(lèi)。
上面的情況下類(lèi)可以正常運(yùn)行。但是,不能用修改過(guò)的接口重新編譯,但是用老的接口編譯仍然可以運(yùn)行。接下來(lái)
- 修改那個(gè)含有 abstract 方法 m() 的接口,創(chuàng)建一個(gè) default 實(shí)現(xiàn)。
- 編譯修改后的接口
- 運(yùn)行類(lèi):失敗。
當(dāng)兩個(gè)接口給同一個(gè)方法都提供了default實(shí)現(xiàn)的時(shí)候,這個(gè)方法是無(wú)法被調(diào)用的,除非實(shí)現(xiàn)類(lèi)也實(shí)現(xiàn)了這個(gè)default方法(要么是直接實(shí)現(xiàn),要么是繼承樹(shù)上更上層的類(lèi)做實(shí)現(xiàn))。
但是,這個(gè)類(lèi)是兼容的。它可以在使用新接口的情況下被載入,甚至可以執(zhí)行,只要它沒(méi)有調(diào)用在兩個(gè)接口中都有 default 實(shí)現(xiàn)的方法。
實(shí)例代碼
為了演示上面的例子,我給 C.java 創(chuàng)建了一個(gè)測(cè)試目錄,它下面還有3個(gè)子目錄,用于存放 I1.java 和 I2.java 。測(cè)試目錄下包含了類(lèi)C的源碼 C.java 。base目錄包含了可以編譯和運(yùn)行的那個(gè)版本的接口。I1包含了有 default 實(shí)現(xiàn)的 m() 方法, I2 不包含任何方法。
實(shí)現(xiàn)類(lèi)包含了 main 方法,所以我們可以在測(cè)試中執(zhí)行它。它會(huì)檢查是否存在命令行參數(shù),這樣,我們就可以很方便的執(zhí)行調(diào)用 m() 和不調(diào)用 m() 的測(cè)試。
public class C implements I1, I2 { public static void main(String[] args) { C c = new C(); if(args.length == 0 ){ c.m(); } } } public interface I1 { default void m(){ System.out.println("hello interface 1"); } } public interface I2 { }
使用下面的命令行來(lái)編譯運(yùn)行:
javac -cp .:base C.java java -cp .:base C hello interface 1
compatible 目錄包含了有 abstract 方法 m() 的 I2 接口,和未修改的 I1 接口。
public interface I2 { void m(); }
這個(gè)不能用來(lái)編譯類(lèi)C:
javac -cp .:compatible C.java C.java:1: error: C is not abstract and does not override abstract method m() in I2 public class C implements I1, I2 { ^ 1 error
錯(cuò)誤信息非常精確。因?yàn)槲覀冇星耙淮尉幾g獲得的 C.class ,如果我們編譯 compatible 目錄下的接口,我們?nèi)匀粫?huì)得到能運(yùn)行實(shí)現(xiàn)類(lèi)的兩個(gè)接口:
javac compatible/I*.java java -cp .:compatible C hello interface 1
第三個(gè)叫做 wrong 的目錄,包含的 I2 接口也定義了 m() 方法:
public interface I2 { default void m(){ System.out.println("hello interface 2"); } }
我們應(yīng)該不厭其煩的編譯它。盡管m()方法被定義了兩次,但是,實(shí)現(xiàn)類(lèi)仍然可以運(yùn)行,只要它沒(méi)有調(diào)用那個(gè)定義了多次的方法,但是,只要我們調(diào)用m()方法,立即就會(huì)失敗。這是我們使用的命令行參數(shù):
javac wrong/*.java java -cp .:wrong C Exception in thread "main" java.lang.IncompatibleClassChangeError: Conflicting default methods: I1.m I2.m at C.m(C.java) at C.main(C.java:5) java -cp .:wrong C x
結(jié)論
當(dāng)你把給接口添加了 default 實(shí)現(xiàn)的類(lèi)庫(kù)移植到Java 8環(huán)境下的時(shí)候,一般不會(huì)有問(wèn)題。至少Java8類(lèi)庫(kù)開(kāi)發(fā)者給集合類(lèi)添加default方法的時(shí)候就是這么想的。使用你類(lèi)庫(kù)的應(yīng)用程序仍然依賴(lài)沒(méi)有 default 方法的Java7的類(lèi)庫(kù)。當(dāng)使用和修改多個(gè)不同的類(lèi)庫(kù)的時(shí)候,有很小的幾率會(huì)發(fā)生沖突。如何才能避免呢?
像以前那樣設(shè)計(jì)你的類(lèi)庫(kù)。可能依賴(lài) default 方法的時(shí)候不要掉以輕心。萬(wàn)不得已不要使用。明智的選擇方法名,避免和其它接口產(chǎn)生沖突。我們將會(huì)學(xué)習(xí)到Java編程中如何使用這個(gè)特性做開(kāi)發(fā)。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
- Java map.getOrDefault()方法的用法詳解
- python字典setdefault方法和get方法使用實(shí)例
- python 字典 setdefault()和get()方法比較詳解
- Python使用defaultdict讀取文件各列的方法
- Python中字典的setdefault()方法教程
- python錯(cuò)誤:AttributeError: ''module'' object has no attribute ''setdefaultencoding''問(wèn)題的解決方法
- Java8新特性之默認(rèn)方法(default)淺析
- default怎么修飾接口中的方法詳解
相關(guān)文章
Java線(xiàn)程中的ThreadLocal詳細(xì)分析
這篇文章主要介紹了Java線(xiàn)程中的ThreadLocal詳細(xì)分析,ThreadLocal?提供線(xiàn)程的局部變量,每個(gè)線(xiàn)程都可以通過(guò)?get()和?set()對(duì)局部變量進(jìn)行操作而不會(huì)對(duì)其他線(xiàn)程的局部變量產(chǎn)生影響,實(shí)現(xiàn)了線(xiàn)程之間的數(shù)據(jù)隔離,需要的朋友可以參考下2023-09-09基于Retrofit+Rxjava實(shí)現(xiàn)帶進(jìn)度顯示的下載文件
這篇文章主要為大家詳細(xì)介紹了基于Retrofit+Rxjava實(shí)現(xiàn)帶進(jìn)度顯示的下載文件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05SpringBoot整合Solr實(shí)現(xiàn)文檔檢索
Solr高度可靠、可擴(kuò)展和容錯(cuò),提供分布式索引、復(fù)制和負(fù)載平衡查詢(xún)、自動(dòng)故障轉(zhuǎn)移和恢復(fù)、集中配置等,Solr 為世界上許多最大的 Internet 站點(diǎn)的搜索和導(dǎo)航功能提供支持,本文將給大家介紹SpringBoot整合Solr實(shí)現(xiàn)文檔檢索,需要的朋友可以參考下2023-08-08java 實(shí)現(xiàn)多個(gè)list 合并成一個(gè)去掉重復(fù)的案例
這篇文章主要介紹了java 實(shí)現(xiàn)多個(gè)list 合并成一個(gè)去掉重復(fù)的案例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08SpringBoot上傳圖片與視頻不顯示問(wèn)題的解決方案
這篇文章主要介紹了關(guān)于springboot上傳圖片與視頻不顯示問(wèn)題,最近做畢設(shè)時(shí)候需要上傳視頻的圖片與視頻,但是每次都需要重啟前端才能展示出此圖片,所以本文給大家介紹了SpringBoot上傳圖片與視頻不顯示問(wèn)題的解決方案,需要的朋友可以參考下2024-03-03JAVA對(duì)字符串進(jìn)行32位MD5加密的實(shí)踐
本文主要介紹了JAVA對(duì)字符串進(jìn)行32位MD5加密的實(shí)踐,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08通過(guò)第三方接口發(fā)送短信驗(yàn)證碼/短信通知(推薦)
這篇文章主要介紹了通過(guò)第三方接口發(fā)送短信驗(yàn)證碼/短信通知(推薦)的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-08-08Spring定時(shí)任務(wù)無(wú)故停止又不報(bào)錯(cuò)的解決
這篇文章主要介紹了Spring定時(shí)任務(wù)無(wú)故停止又不報(bào)錯(cuò)的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11idea已經(jīng)提交到遠(yuǎn)程分支,但需要本地和遠(yuǎn)程都回退到某一版本問(wèn)題
這篇文章主要介紹了idea已經(jīng)提交到遠(yuǎn)程分支,但需要本地和遠(yuǎn)程都回退到某一版本問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11java swing實(shí)現(xiàn)貪吃蛇雙人游戲
這篇文章主要為大家詳細(xì)介紹了java swing實(shí)現(xiàn)貪吃蛇雙人小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-01-01