謹(jǐn)慎使用Java8的默認(rèn)方法
默認(rèn)方法給JVM的指令集增加了一個(gè)非常不錯(cuò)的新特性。使用了默認(rèn)方法之后,如果庫中的接口增加了新的方法,實(shí)現(xiàn)了這個(gè)接口的用戶類能夠自動獲得這個(gè)方法的默認(rèn)實(shí)現(xiàn)。一旦用戶想更新他的實(shí)現(xiàn)類的話,只需覆蓋一下這個(gè)默認(rèn)方法就可以了,取而代之的是一個(gè)在特定場景下更有意義的實(shí)現(xiàn)。更棒的是,用戶可以在重寫的方法里面調(diào)用接口的默認(rèn)實(shí)現(xiàn)來增加一些額外的功能。
目前為止一切都還不錯(cuò)。然而,給現(xiàn)有的Java接口增加默認(rèn)方法可能會導(dǎo)致代碼的不兼容??磦€(gè)例子就很容易能明白了。假設(shè)有一個(gè)庫,它需要用戶實(shí)現(xiàn)它的一個(gè)接口作為輸入:
interface SimpleInput { void foo(); void bar(); } abstract class SimpleInputAdapter implements SimpleInput { @Override public void bar() { // some default behavior ... } }
在Java 8以前,上述這種接口和一個(gè)對應(yīng)的適配器類的組合在Java語言中是一種很常見的模式。類庫的開發(fā)人員提供了一個(gè)適配器來減少庫使用者的編碼量。然而提供這個(gè)接口的目的其實(shí)是為了能實(shí)現(xiàn)某種類似多重繼承的關(guān)系。
我們假設(shè)有一個(gè)用戶使用了這個(gè)適配器:
class MyInput extends SimpleInputAdapter{ @Override public void foo() { // do something ... } @Override public void bar() { super.bar(); // do something additionally ... } }
有了這個(gè)實(shí)現(xiàn),用戶可以和庫進(jìn)行交互了。注意這個(gè)實(shí)現(xiàn)是如何重寫bar方法來給默認(rèn)的實(shí)現(xiàn)增加額外的功能的。
那如果這個(gè)庫遷移到Java 8的話會怎樣?首先,這個(gè)庫很可能會廢棄掉這個(gè)適配器類并將這個(gè)功能遷移到默認(rèn)方法里。最終這個(gè)接口看起來會是這樣的:
interface SimpleInput { void foo(); default void bar() { // some default behavior } }}
有了這個(gè)新接口后,用戶得更新他的代碼來使用這個(gè)默認(rèn)方法,而不再是適配器類了。使用新接口而非適配器類的一大好處就是,用戶可以去繼承一個(gè)別的類而不是這個(gè)適配器類了。我們來動手實(shí)踐一下,將MyInput類改造成使用默認(rèn)方法。由于現(xiàn)在我們可以繼承別的類了,我們再額外地?cái)U(kuò)展一個(gè)第三方的基類試試。這個(gè)基類具體是做什么的在這里并不重要,我們先假設(shè)一下這么做對我們這個(gè)用例來說是有意義的。
class MyInput extends ThirdPartyBaseClass implements SimpleInput { @Override public void foo() { // do something ... } @Override public void bar() { SimpleInput.super.foo(); // do something additionally ... } }
為了實(shí)現(xiàn)和原先那個(gè)類同樣的功能,這里我們用到了Java 8的新語法來調(diào)用接口的默認(rèn)方法。同樣的,我們把myMethod的邏輯放到某個(gè)基類MyBase里面。可以捶捶肩膀放松下了。重構(gòu)之后棒極了!
我們使用的這個(gè)庫得到了很大的改進(jìn)。然而,維護(hù)人員需要添加另一個(gè)接口來實(shí)現(xiàn)一些額外的功能。這個(gè)接口叫做CompexInput ,它繼承了SimpleInput類,并增加了一個(gè)額外的方法。由于通常都認(rèn)為默認(rèn)方法是可以放心地添加的,因此維護(hù)人員重寫了SimpleInput類的默認(rèn)方法并添加了一些額外的動作來給用戶提供一個(gè)更好的默認(rèn)實(shí)現(xiàn)。畢竟使用適配器類的時(shí)候這個(gè)做法也十分常見:
interface ComplexInput extends SimpleInput { void qux(); @Override default void bar() { SimpleInput.super.bar(); // so complex, we need to do more ... } }
這個(gè)新特性看起來非常不錯(cuò),因此ThirdPartyBaseClass類的維護(hù)人員也決定使用這個(gè)庫了。為了實(shí)現(xiàn)這個(gè),他將ThirdPartyBaseClass類實(shí)現(xiàn)了ComplexInput接口。
但這樣的話對MyInput類意味著什么?由于它繼承了ThirdPartyBaseClass類,因此默認(rèn)實(shí)現(xiàn)了ComplexInput接口,這樣的話調(diào)用SimpleInput的默認(rèn)方法就不合法了。結(jié)果就是,用戶的代碼最后無法通過編譯。還有就是,現(xiàn)在已經(jīng)徹底無法調(diào)用這個(gè)方法了,因?yàn)镴ava把這種調(diào)用間接父類的super-super方法認(rèn)為是不合法的。你只能去調(diào)用ComplexInput接口的默認(rèn)方法了。然而這首先需要你在MyInput類中顯式的實(shí)現(xiàn)一下這個(gè)接口。對于這個(gè)庫的用戶而言,這些改動完全是意想不到的。
(注:簡單點(diǎn)說其實(shí)就是:
interface A { default void test() { } } interface B extends A { default void test() { } } public class Test implements B { public void test() { B.super.test(); //A.super.test(); 錯(cuò)誤 } }
當(dāng)然這么寫的話是用戶主動選擇實(shí)現(xiàn)了B接口,而文中的例子由于引入了一個(gè)基類,因此由于庫和基類中都進(jìn)行了一個(gè)看似沒有影響的改動,實(shí)際上卻導(dǎo)致用戶代碼無法通過編譯)
很奇怪的是,Java在運(yùn)行時(shí)并沒有對這個(gè)進(jìn)行區(qū)分。JVM的校驗(yàn)器允許一個(gè)編譯過的類進(jìn)行SimpleInput::foo方法的調(diào)用,盡管加載的這個(gè)類繼承了ThirdPartyBaseClass的更新版本后隱式地實(shí)現(xiàn)了ComplexInput接口。要怪只能怪編譯器了。(注:編譯器與運(yùn)行時(shí)的行為不一致)
那我們從中學(xué)到了什么?簡單地說,不要在另一個(gè)接口中重寫原接口的默認(rèn)方法。不要用另一個(gè)默認(rèn)方法來重寫它,也不要某個(gè)抽象方法來重寫它??偠灾褂媚J(rèn)方法時(shí)應(yīng)當(dāng)十分謹(jǐn)慎。雖然它們使得Java現(xiàn)有的集合庫的接口更容易改進(jìn)了,但它允許你在類的繼承結(jié)構(gòu)中進(jìn)行方法調(diào)用,這本質(zhì)上其實(shí)是增加了復(fù)雜性。在Java 7以前,你只需遍歷線性的類層次結(jié)構(gòu)看一下實(shí)際調(diào)用的代碼就可以了。當(dāng)你覺得的確需要的時(shí)候,再去使用默認(rèn)方法。
以上就是針對為什么要慎用Java8的默認(rèn)方法進(jìn)行的詳細(xì)解釋,希望對大家的學(xué)習(xí)有所幫助。
- Java8接口的默認(rèn)方法
- Java8新特性之默認(rèn)方法(default)淺析
- Java8中新特性O(shè)ptional、接口中默認(rèn)方法和靜態(tài)方法詳解
- 一篇文章帶你認(rèn)識Java8接口的默認(rèn)方法
- 30分鐘入門Java8之默認(rèn)方法和靜態(tài)接口方法學(xué)習(xí)
- Java8默認(rèn)方法Default Methods原理及實(shí)例詳解
- java8新特性之接口默認(rèn)方法示例詳解
- Java8中的默認(rèn)方法(面試者必看)
- Java8新特性之默認(rèn)方法和靜態(tài)方法
- Java8新特性之默認(rèn)方法詳解
相關(guān)文章
Java使用iTextPDF生成PDF文件的實(shí)現(xiàn)方法
這篇文章主要介紹了Java使用iTextPDF生成PDF文件的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02記一次springboot服務(wù)凌晨無故宕機(jī)問題的解決
這篇文章主要介紹了記一次springboot服務(wù)凌晨無故宕機(jī)問題的解決,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09SpringBoot項(xiàng)目網(wǎng)頁加載出現(xiàn)Whitelabel?Error?Page的解決
這篇文章主要介紹了SpringBoot項(xiàng)目網(wǎng)頁加載出現(xiàn)Whitelabel?Error?Page的解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11Java調(diào)用dll文件的實(shí)現(xiàn)解析
這篇文章主要介紹了Java調(diào)用dll文件的實(shí)現(xiàn)解析,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02Spring Boot 啟動、停止、重啟、狀態(tài)腳本
今天給大家分享Spring Boot 項(xiàng)目腳本(啟動、停止、重啟、狀態(tài)),通過示例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2021-06-06Spring中bean的初始化和銷毀幾種實(shí)現(xiàn)方式詳解
這篇文章主要介紹了Spring中bean的初始化和銷毀幾種實(shí)現(xiàn)方式詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11Java 判斷字符串a(chǎn)和b是否互為旋轉(zhuǎn)詞
本篇文章主要介紹了判斷字符串a(chǎn)和b是否互為旋轉(zhuǎn)詞的相關(guān)知識,具有很好的參考價(jià)值。下面跟著小編一起來看下吧2017-05-05Idea創(chuàng)建多模塊maven聚合項(xiàng)目的實(shí)現(xiàn)
這篇文章主要介紹了Idea創(chuàng)建多模塊maven聚合項(xiàng)目的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12