我用 Java 8 寫了一段邏輯,同事直呼看不懂,你試試看
前言
用Java 8 寫了一段邏輯,同事居然說看不懂,以下是業務背景,大家可以一起看看!
業務背景
首先,業務需求是這樣的,從第三方電商平臺拉取所有訂單,然后保存到公司自己的數據庫,需要判斷是否有物流信息,如果有物流信息,還需要再進行上傳。
而第三方接口返回的數據是 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 來實現了:
- /**
- * /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 了,之前也陸續分享了一系列新特性文章。
所以我現在雖然是個老前浪了,但在新知識學習和掌握上面,我感覺已經走到了很多后浪前面。




























