作者丨一書生VOID lw900925.github.io/java/java8-optional.html Optional 類是 Java 8 新增的一個(gè)類,用以解決程序中常見的NullPointerException 異常問題。本篇文章將詳細(xì)介紹 Optional 類,以及如何用它消除代碼中的 null 檢查。 避免使用null檢查作為Java開發(fā)人員,幾乎所有人都遇到過NullPointerException異常,大多數(shù)人遇到NullPointerException異常時(shí)都會在異常出現(xiàn)的地方加上if代碼塊來判斷值不為空,比如下面的代碼:
這是比較普遍的做法,為了避免出現(xiàn)NullPointerException異常,手動對可能為null值進(jìn)行了處理,不過代碼看起來非常糟糕,業(yè)務(wù)邏輯被淹沒在if邏輯判斷中,也許下面的代碼看起來可讀性稍好一些: public String bindUserToRole(User user) { 上面的代碼避免了深層的if語句嵌套,但本質(zhì)上是一樣的,方法內(nèi)有三個(gè)不同的返回點(diǎn),出錯后調(diào)試也不容易,因?yàn)槟悴恢朗悄莻€(gè)值導(dǎo)致了NullPointerException異常。 基于上面的原因,Java 8中引入了一個(gè)新的類Optional,用以避免使用null值引發(fā)的種種問題。擴(kuò)展:如何更優(yōu)雅的處理空值? Optional類
創(chuàng)建Optional對象Optional類提供類三個(gè)方法用于實(shí)例化一個(gè)Optional對象,它們分別為empty()、of()、ofNullable(),這三個(gè)方法都是靜態(tài)方法,可以直接調(diào)用。 empty()方法用于創(chuàng)建一個(gè)沒有值的Optional對象:
empty()方法創(chuàng)建的對象沒有值,如果對emptyOpt變量調(diào)用isPresent()方法會返回false,調(diào)用get()方法拋出NullPointerException異常。 of()方法使用一個(gè)非空的值創(chuàng)建Optional對象: String str = 'Hello World'; ofNullable()方法接收一個(gè)可以為null的值:
如果str的值為null,得到的nullableOpt是一個(gè)沒有值的Optional對象。 提取Optional對象中的值如果我們要獲取User對象中的roleId屬性值,常見的方式是直接獲?。?/p> String roleId = null; 使用Optional中提供的map()方法可以以更簡單的方式實(shí)現(xiàn):
使用orElse()方法獲取值Optional類還包含其他方法用于獲取值,這些方法分別為:
下面來看看這三個(gè)方法的具體用法: String str = 'Hello World'; 此外,Optional類還提供了一個(gè)ifPresent()方法,該方法接收一個(gè)Consumer<? super T>函數(shù)式接口,一般用于將信息打印到控制臺:
使用filter()方法過濾filter()方法可用于判斷Optional對象是否滿足給定條件,一般用于條件過濾: Optional<String> optional = Optional.of('lw900925@163.com'); 在上面的代碼中,如果filter()方法中的Lambda表達(dá)式成立,filter()方法會返回當(dāng)前Optional對象值,否則,返回一個(gè)值為空的Optional對象。Java知音公眾號內(nèi)回復(fù)“后端面試”,送你一份Java面試題寶典。 如何正確使用Optional通過上面的例子可以看出,Optional類可以優(yōu)雅的避免NullPointerException帶來的各種問題,不過,你是否真正掌握了Optional的用法? 假設(shè)你試圖使用Optional來避免可能出現(xiàn)的NullPointerException異常,編寫了如下代碼:
坦白說,上面的代碼與我們之前的使用if語句判斷空值沒有任何區(qū)別,沒有起到Optional的正真作用: if (user != null) { 當(dāng)我們從之前版本切換到Java 8的時(shí)候,不應(yīng)該還按照之前的思維方式處理null值,Java 8提倡函數(shù)式編程,新增的許多API都可以用函數(shù)式編程表示,Optional類也是其中之一。這里有幾條關(guān)于Optional使用的建議:
第一條建議中直接調(diào)用get()方法是很危險(xiǎn)的做法,如果Optional的值為空,那么毫無疑問會拋出NullPointerException異常,而為了調(diào)用get()方法而使用isPresent()方法作為空值檢查,這種做法與傳統(tǒng)的用if語句塊做空值檢查沒有任何區(qū)別。 第二條建議避免使用Optional作為實(shí)體類的屬性,它在設(shè)計(jì)的時(shí)候就沒有考慮過用來作為類的屬性,如果你查看Optional的源代碼,你會發(fā)現(xiàn)它沒有實(shí)現(xiàn)java.io.Serializable接口,這在某些情況下是很重要的(比如你的項(xiàng)目中使用了某些序列化框架),使用了Optional作為實(shí)體類的屬性,意味著他們不能被序列化。 下面我們通過一些例子講解Optional的正確用法: 正確創(chuàng)建Optional對象上面提到創(chuàng)建Optional對象有三個(gè)方法,empty()方法比較簡單,沒什么特別要說明的。主要是of()和ofNullable()方法。當(dāng)你很確定一個(gè)對象不可能為null的時(shí)候,應(yīng)該使用of()方法,否則,盡可能使用ofNullable()方法,比如:
orElse()方法的使用return str != null ? str : 'Hello World' 上面的代碼表示判斷字符串str是否為空,不為空就返回,否則,返回一個(gè)常量。使用Optional類可以表示為:
簡化if-elseUser user = ... 上面的代碼可以簡化成:
總結(jié)一下,新的Optional類讓我們可以以函數(shù)式編程的方式處理null值,拋棄了Java 8之前需要嵌套大量if-else代碼塊,使代碼可讀性有了很大的提高。 |
|