前言
創(chuàng)新互聯(lián)主要從事網(wǎng)站制作、做網(wǎng)站、網(wǎng)頁設(shè)計(jì)、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)香河,10年網(wǎng)站建設(shè)經(jīng)驗(yàn),價(jià)格優(yōu)惠、服務(wù)專業(yè),歡迎來電咨詢建站服務(wù):18980820575
要說 Java 編程中哪個(gè)異常是你印象最深刻的,那 NullPointerException 空指針可以說是臭名昭著的。不要說初級程序員會碰到, 即使是中級,專家級程序員稍不留神,就會掉入這個(gè)坑里。
Null 引用的發(fā)明者Tony Hoare 曾在 2009 年作出道歉聲明,聲明中表示,到目前為止,空指針異常大約給企業(yè)已造成數(shù)十億美元的損失。
下面是 Tony Hoare 的原話:
我將 Null 引用的設(shè)計(jì)稱為是一個(gè)數(shù)十億美元的錯(cuò)誤。1965 那年,我正在用面向?qū)ο笳Z言(ALGOL W) 設(shè)計(jì)首個(gè)功能全面的系統(tǒng)。當(dāng)時(shí)我的考量是,確保所有被使用的引用都是安全的,編譯器會自動進(jìn)行檢查。但是,我沒有抵住誘惑,加入了 Null 引用,僅僅是為了實(shí)現(xiàn)起來省事。這之后,它導(dǎo)致了數(shù)不清的 bug、錯(cuò)誤和系統(tǒng)崩潰,也為企業(yè)導(dǎo)致了不可估量的損失。
事已至此,我們必須學(xué)會面對它。So, 我們要如何防止空指針異常呢?
唯一的辦法就是對可能為 Null 的對象添加檢查。但是 Null 檢查是繁瑣且痛苦的。所以一些比較新的語言為了處理 Null 檢查,特意添加了特殊的語法,如空合并運(yùn)算符。
在Groovy 或Kotlin 這樣的語言中也被稱為 Elvis 運(yùn)算符。
不幸的是,在老版本的 Java 中并沒有提供這樣的語法糖。Java8 中在這方面做了改進(jìn)。所以,這篇文章就特意來介紹一下如何在 Java8 中利用新特性來編寫防止 NullPointerException 的發(fā)生。
Java8 中如何加強(qiáng)對 Null 對象的檢查?
在上篇文章 Java8 新特性指導(dǎo)手冊 中簡單的提了一下如何通過 Optional 類來對對象做空校驗(yàn)。接下來,我們再細(xì)說一下:
在業(yè)務(wù)系統(tǒng)中,對象中嵌套對象是經(jīng)常發(fā)生的場景,如下示例代碼:
// 最外層對象 class Outer { Nested nested; Nested getNested() { return nested; } } // 第二層對象 class Nested { Inner inner; Inner getInner() { return inner; } } // 最底層對象 class Inner { String foo; String getFoo() { return foo; } }
業(yè)務(wù)中,假設(shè)我們需要獲取 Outer 對象對底層的 Inner 中的 foo 屬性,我們必須寫一堆的非空校驗(yàn),來防止發(fā)生 NullPointerException :
// 繁瑣的代碼 Outer outer = new Outer(); if (outer != null && outer.nested != null && outer.nested.inner != null) { System.out.println(outer.nested.inner.foo); }
通過 Optional
在 Java8 中,我們有更優(yōu)雅的解決方式,那就是使用 Optional 是說,我們可以在一行代碼中,進(jìn)行流水式的 map 操作。而 map 方法內(nèi)部會自動進(jìn)行空校驗(yàn) :
Optional.of(new Outer()) .map(Outer::getNested) .map(Nested::getInner) .map(Inner::getFoo .ifPresent(System.out::println); // 如果不為空,最終輸出 foo 的值
通過 suppiler 函數(shù)自定義增強(qiáng) API
上面這種方式個(gè)人感覺還是有點(diǎn)啰嗦,我們可以利用 suppiler 函數(shù)來出一個(gè)終極解決方案:
public staticOptional resolve(Supplier resolver) { try { T result = resolver.get(); return Optional.ofNullable(result); } catch (NullPointerException e) { // 可能會拋出空指針異常,直接返回一個(gè)空的 Optional 對象 return Optional.empty(); } }
利用上面的 resolve 方法來重構(gòu)上述的非空校驗(yàn)代碼段:
Outer obj = new Outer(); // 直接調(diào)用 resolve 方法,內(nèi)部做空指針的處理 resolve(() -> obj.getNested().getInner().getFoo()); .ifPresent(System.out::println); // 如果不為空,最終輸出 foo 的值
最后
你需要知道的是,上面這兩個(gè)解決方案并沒傳統(tǒng)的 null 檢查性能那么高效。但在絕大部分業(yè)務(wù)場景下,舍棄那么一丟丟的性能來方便編碼,是完全可取,除非是那種對性能有嚴(yán)格要求的,我們才不建議使用。
個(gè)人覺得,真要拿這點(diǎn)性能說事,還不如去優(yōu)化優(yōu)化 sql 語句,業(yè)務(wù)邏輯等。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對創(chuàng)新互聯(lián)的支持。