我用 Java 8 寫了一段邏輯,同事直呼看不懂,你試試看。。

Java技術棧發表於2020-09-03

業務背景

首先,業務需求是這樣的,從第三方電商平臺拉取所有訂單,然後儲存到公司自己的資料庫,需要判斷是否有物流資訊,如果有物流資訊,還需要再進行上傳。

而第三方介面返回的資料是 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 程式猿要學的技術很多,雖然有點知識點短時間你是理解了,但肯定不深刻,時間久了就忘了,所以給大家的建議是一定要實戰 + 閱讀原始碼,這樣才真正屬於你的。

覺得不錯,在看、轉發支援下哦~

推薦去我的部落格閱讀更多:

1.Java JVM、集合、多執行緒、新特性系列教程

2.Spring MVC、Spring Boot、Spring Cloud 系列教程

3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程

4.Java、後端、架構、阿里巴巴等大廠最新面試題

覺得不錯,別忘了點贊+轉發哦!

相關文章