Java8新特性之Optional的應用--多層對象取值



轉載自:http://www.javastack.cn/article/2020/java-8-optional-map-do-you-know/


業務背景

首先,業務需求是這樣的,從第三方電商平台拉取所有訂單,然后保存到公司自己的數據庫,需要判斷是否有物流信息,如果有物流信息,還需要再進行上傳。

而第三方接口返回的數據是 JSON 格式的,其中物流信息卻藏的十分深,如下面所示,JSON 節點是這樣的:

xxxOrder > xxxShippingInfo > xxxShipmentDetails > xxxTrackingInfo > trackingNumber, trackingLink

基本實現

因為第三方接口返回的數據是 JSON 格式的,所以需要把 JSON 字符串轉換成 Java 對象來進行處理。

@JsonIgnoreProperties(ignoreUnknown = true)
public class XxxOrder {

/**
* 物流信息
*/
@JsonProperty("shippingInfo")
private XxxShippingInfo xxxShippingInfo;

}

上面只是第一層示例,要拿到物流信息,要依次封裝四層對象,到真正獲取物流信息時要避免空指針,就需要判斷四層才能拿到,如示例所示:

if(xxxOrder != null){
if(xxxOrder.getXxxShippingInfo() != null){
if(xxxOrder.getXxxShippingInfo().getXxxShipmentDetails() != null){
if(xxxOrder.getXxxShippingInfo().getXxxShipmentDetails().getXxxTrackingInfo() != null){
...
}
}
}
}

獲取一個物流信息這么麻煩,我也是醉了,這樣寫也太不優雅了。

Java 8 實現

因為我知道 Java 8 可以處理這類的需求,所以我從來沒想過用最原始的方式去實現,直接把就用 Java 8 來實現了:

/**
* 公眾號:Java技術棧
/
private String[] getFulfillments(XxxOrder xxxOrder) {
return Optional.ofNullable(xxxOrder)
.map((o) -> o.getXxxShippingInfo())
.map((si) -> si.getXxxShipmentDetails())
.map((sd) -> sd.getXxxTrackingInfo())
.map((t) -> new String[]{t.getTrackingNumber(), t.getTrackingLink()})
.orElse(null);
}

寫完之后,同事居然都直呼看不懂,還特地跑過來問我。。

實現原理

其實這並沒有用什么高超的技術,就是利用 Java 8 Optional 來實現的,細節就不介紹了 ,主要是為了避免空指針而生的,不懂的可以點擊這里查看這篇文章。

今天就來介紹下 Optional#map 方法實現這段邏輯的原理,來看下 map 的實現源碼:

public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
// 函數式接口不能為null
Objects.requireNonNull(mapper);

// 如果當前沒有值,返回一個空的Optional
if (!isPresent())
return empty();
else {
// 如果當前有值,返回一個函數式處理該值的結果Optional
return Optional.ofNullable(mapper.apply(value));
}
}

// 判斷 Optional Value 有沒有值
public boolean isPresent() {
return value != null;
}

// 創建一個 Optional,可以為空
public static <T> Optional<T> ofNullable(T value) {
return value == null ? empty() : of(value);
}

所以回到這段程序:

// 根對象為空就創建一個空Optional,否則就創建一個根對象的Optional
Optional.ofNullable(xxxOrder)
// 根對象為空就直接返回空Optional,否則返回這個值的 Optional
.map((o) -> o.getXxxShippingInfo())
// 下面依次類推……
.map((si) -> si.getXxxShipmentDetails())
.map((sd) -> sd.getXxxTrackingInfo())
.map((t) -> new String[]{t.getTrackingNumber(), t.getTrackingLink()})
// 取不到值就返回 null
.orElse(null);
}

也許你看完感覺還是看不懂,我承認,確實比較繞,不太好理解,這個只可意會不可言傳了,多看多練就理解了。

這個的關鍵核心在於,調用 map 時,如果 Optional 沒有值就直接返回空的 Optional,而不會調用函數式接口,所以就不會出現空指針。所以只要有一個為空,后面就取不到物流信息。

程序使用了 .xx.xx.xx 這樣的鏈式調用,調用 map 方法就必須是 Optional,而 map 的返回結果就是 Optional。

有一個問題是,如果都為空,那不是所有的 map 都會走一遍?在這種情況下會不會影響性能?編譯器是否會作優化?這個暫不可知。

另外還有一個 flatMap 方法,和 map 有什么區別呢?

flatMap 返回結果需要在函數式接口中封裝 Optional 返回,在這里應用不太合適。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM