API
1. empty()
返回一個Optional
容器對象绅络,而不是 null。
2. of(T value)
創(chuàng)建一個Optional
對象恼策,如果 value 是 null码倦,則拋出 NPE企孩。
3. ofNullable(T value)
同上,創(chuàng)建一個Optional
對象袁稽,但 value 為空時返回Optional.empty()
勿璃。
4. get()
返回Optional
中包裝的值,在判空之前推汽,千萬不要直接使用补疑!
5. orElse(T other)
同樣是返回Optional
中包裝的值,但不同的是當取不到值時歹撒,返回你指定的 default莲组。
6. orElseGet(Supplier<? extends T> other)
同樣是返回Optional
中包裝的值,取不到值時暖夭,返回你指定的 default锹杈。
7. orElseThrow(Supplier<? extends X> exceptionSupplier)
返回Optional
中包裝的值,取不到值時拋出指定的異常迈着。
8. isPresent()
判斷Optional
中是否有值竭望,返回 boolean,某些情況下很有用寥假,但盡量不要用在 if 判斷體中市框。
9. ifPresent(Consumer<? super T> consumer)
判斷Optional
中是否有值,有值則執(zhí)行 consumer糕韧,否則什么都不干枫振。
10. filter(Predicate<? super T> predicate)
過濾Optional
中的值,斷言通過返回原值萤彩,否則返回empty()粪滤。
11. map(Function<? super T, ? extend U> mapper)
轉換Optional
中的值。
12. flatMap(Function<? super T, Optional<U>> mapper)
轉換Optional
中的值雀扶,與map的區(qū)別是杖小,flatMap不會自動包裝結果,需要mapper返回Optional結果愚墓。
最佳實踐
首先是一些基本原則:
不要聲明任何
Optional
實例屬性不要在任何 setter 或者構造方法中使用
Optional
Optional
屬于返回類型予权,在業(yè)務返回值或者遠程調用中使用
1. 業(yè)務上需要空值時,不要直接返回 null浪册,使用**Optional.empty()**
public Optional<User> getUser(String name) {
if (StringUtil.isNotEmpty(name)) {
return RemoteService.getUser(name);
}
return Optional.empty();
}
2. 使用 orElseGet()
獲取 value 有三種方式:get()
orElse()
orElseGet()
扫腺。這里推薦在需要用到的地方只用 orElseGet()
。
首先村象,get()
不能直接使用笆环,需要結合判空使用攒至。這和!=null
其實沒多大區(qū)別,只是在表達和抽象上有所改善躁劣。
其次迫吐,為什么不推薦orElse()
呢?因為orElse()
無論如何都會執(zhí)行括號中的內(nèi)容账忘, orElseGet()
只在主體 value 是空時執(zhí)行志膀,下面看個例子:
public String getName() {
System.out.print("method called");
}
String name1 = Optional.of("String").orElse(getName()); //output: method called
String name2 = Optional.of("String").orElseGet(() -> getName()); //output:
如果上面的例子getName()
方法是一個遠程調用,或者涉及大量的文件 IO闪萄,代價可想而知梧却。
但 orElse()
就一無是處嗎?并不是败去。orElseGet()
需要構建一個Supplier
,如果只是簡單的返回一個靜態(tài)資源烈拒、字符串等等圆裕,直接返回靜態(tài)資源即可。
public static final String USER_STATUS = "UNKNOWN";
...
public String findUserStatus(long id) {
Optional<String> status = ... ; //
return status.orElse(USER_STATUS);
}
//不要這么寫
public String findUserStatus(long id) {
Optional<String> status = ... ; //
return status.orElse("UNKNOWN");//這樣每次都會新建一個String對象
}
3. 使用 orElseThrow()
這個針對阻塞性的業(yè)務場景比較合適荆几,例如沒有從上游獲取到用戶信息吓妆,下面的所有操作都無法進行,那此時就應該拋出異常吨铸。正常的寫法是先判空行拢,再手動 throw 異常,現(xiàn)在可以集成為一行:
public String findUser(long id) {
Optional<User> user = remoteService.getUserById(id) ;
return user.orElseThrow(IllegalStateException::new);
}
4. 不為空則執(zhí)行時诞吱,使用 ifPresent()
這點沒有性能上的優(yōu)勢舟奠,但可以使代碼更簡潔:
if (status.isPresent()) {
System.out.println("Status: " + status.get());
}
//現(xiàn)在
status.ifPresent(System.out::println);
5. 不要濫用
有些簡單明了的方法,完全沒必要增加Optional
來增加復雜性房维。
public String fetchStatus() {
String status = getStatus() ;
return Optional.ofNullable(status).orElse("PENDING");
}
//判斷一個簡單的狀態(tài)而已
public String fetchStatus() {
String status = ... ;
return status == null ? "PENDING" : status;
}
首先沼瘫,null 可以作為集合的元素之一,它并不是非法的咙俩;其次耿戚,集合類型本身已經(jīng)具備了完整的空表達,再去包裝一層Optional
也是徒增復雜阿趁,收益甚微膜蛔。例如,map 已經(jīng)有了getOrDefault()
這樣的類似orElse()
的 API 了脖阵。
總結
Optional
的出現(xiàn)使 Java 對 null 的表達能力更近了一步皂股,好馬配好鞍,合理使用可以避免大量的 NPE独撇,節(jié)省大量的人力物力屑墨。以上內(nèi)容也是本人查詢了很多資料躁锁,邊學邊寫的產(chǎn)出,如有錯漏之處卵史,還請不吝指教战转。