業務背景
首先,業務需求是這樣的,從第三方電商平台拉取所有訂單,然后保存到公司自己的數據庫,需要判斷是否有物流信息,如果有物流信息,還需要再進行上傳。
而第三方接口返回的數據是 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 返回,在這里應用不太合適。
總結
很多人一直都在說有在學習 Java 8 新特性,但在我看來,大部分人並沒有什么實踐,用的都還是最原始的實現方式。
其實我個人是一直在努力學習這方面的知識的,最新的我已經學到 Java 14 了,之前也陸續分享了一系列新特性文章,感興趣的可以關注公眾號Java技術棧回復java獲取。
所以我現在雖然是個老前浪了,但在新知識學習和掌握上面,我感覺已經走到了很多后浪前面。
做 Java 程序猿要學的技術很多,雖然有點知識點短時間你是理解了,但肯定不深刻,時間久了就忘了,所以給大家的建議是一定要實戰 + 閱讀源碼,這樣才真正屬於你的。
覺得不錯,在看、轉發支持下哦~
推薦去我的博客閱讀更多:
2.Spring MVC、Spring Boot、Spring Cloud 系列教程
3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程
覺得不錯,別忘了點贊+轉發哦!