Reactor中的onErrorContinue?和?onErrorResume
前言
這似乎是 Reactor 的熱門搜索之一,至少當我在谷歌中輸入 onErrorContinue 時,onErrorResume 會在它旁邊彈出。讓我把我的測試代碼和我的一些解釋粘貼在下面。
1 基礎功能
這是一個簡單的函數(shù),將 5 個連續(xù)的數(shù)字分別乘以 2,然后相加,當 i==2 時拋出一個異常:
public static void main(String... args) { Flux.range(1,5) .doOnNext(i -> System.out.println("input=" + i)) .map(i -> i == 2 ? i / 0 : i) .map(i -> i * 2) .reduce((i,j) -> i+j) .doOnNext(i -> System.out.println("sum=" + i)) .block(); }
顯然,輸出如下:
input=1
input=2
Exception in thread "main" java.lang.ArithmeticException: / by zero
2 只有 onErrorResume ()
public static void main(String... args) { Flux.range(1,5) .doOnNext(i -> System.out.println("input=" + i)) .map(i -> i == 2 ? i / 0 : i) .map(i -> i * 2) .onErrorResume(err -> { log.info("onErrorResume"); return Flux.empty(); }) .reduce((i,j) -> i+j) .doOnNext(i -> System.out.println("sum=" + i)) .block(); }
輸出如下:
input=1
input=2
17:40:47.828 [main] INFO com.example.demo.config.TestRunner - onErrorResume
sum=2
正如官方文檔描述(onErrorResume)的那樣,onErrorResume
用返回內容替換 Flux,因此在 2 之后的數(shù)據不會被處理。唯一值得一提的是,onErrorResume()
不必馬上在錯誤之后捕獲異常。在 onErrorContinue()
的官網文檔中(onErrorContinue),只有 onErrorContinue()
強調了了 upstream 關鍵詞,但顯然,它還有其他含義。
3 只有 onErrorContinue()
public static void main(String... args) { Flux.range(1,5) .doOnNext(i -> System.out.println("input=" + i)) .map(i -> i == 2 ? i / 0 : i) .map(i -> i * 2) .onErrorContinue((err, i) -> {log.info("onErrorContinue={}", i);}) .reduce((i,j) -> i+j) .doOnNext(i -> System.out.println("sum=" + i)) .block(); }
輸出:
input=1
input=2
17:43:10.656 [main] INFO com.example.demo.config.TestRunner - onErrorContinue=2
input=3
input=4
input=5
sum=26
顯然,onErrorContinue
會丟棄錯誤數(shù)據 2, 然后繼續(xù)數(shù)字 3 直到 5。
4 onErrorResume() 然后 onErrorContinue()
public static void main(String... args) { Flux.range(1,5) .doOnNext(i -> System.out.println("input=" + i)) .map(i -> i == 2 ? i / 0 : i) .map(i -> i * 2) .onErrorResume(err -> { log.info("onErrorResume"); return Flux.empty(); }) .onErrorContinue((err, i) -> {log.info("onErrorContinue={}", i);}) .reduce((i,j) -> i+j) .doOnNext(i -> System.out.println("sum=" + i)) .block(); }
輸出和上面一樣:
input=1
input=2
17:47:05.789 [main] INFO com.example.demo.config.TestRunner - onErrorContinue=2
input=3
input=4
input=5
sum=26
這樣的結果,你想到了嗎?onErrorContinue()
會在 onErrorResume()
得到錯誤之前處理這個錯誤。當兩個錯誤處理函數(shù)在同一個函數(shù)中的時候很明顯,但是當你的函數(shù)中只有 onErrorResume()
,而一些調用者實際上有 onErrorContinue()
時,你的 onErrorResume()
沒有被調用的原因可能就不那么明顯了。
5 使用 onErrorResume() 模擬 onErrorContinue()
有些博客建議我們完全不用 onErrorContinue()
,且在所有場景中僅用 onErrorResume()
。但是上述示例已經展示了它們會產生不同的結果。那我們怎么實現(xiàn)呢?
public static void main(String... args) { Flux.range(1,5) .doOnNext(i -> System.out.println("input=" + i)) .flatMap(i -> Mono.just(i) .map(j -> j == 2 ? j / 0 : j) .map(j -> j * 2) .onErrorResume(err -> { System.out.println("onErrorResume"); return Mono.empty(); }) ) .reduce((i,j) -> i+j) .doOnNext(i -> System.out.println("sum=" + i)) .block(); }
因此,本質上是將可能在 flatMap 或 concatMap 中拋出錯誤的操作包裝起來,并在其上使用 onErrorResume()
。這樣,它會產生相同的結果:
input=1
input=2
onErrorResume
input=3
input=4
input=5
sum=26
6 使用 onErrorResume() 和下游的 onErrorContinue() 模擬 onErrorContinue()
有時候,onErrorContinue() 放在調用程序中,您無法控制它。但你仍然需要 onErrorResume()。你該怎么辦?
public static void main(String... args) { Flux.range(1,5) .doOnNext(i -> System.out.println("input=" + i)) .flatMap(i -> Mono.just(i) .map(j -> j == 2 ? j / 0 : j) .map(j -> j * 2) .onErrorResume(err -> { System.out.println("onErrorResume"); return Mono.empty(); }) .onErrorStop() ) .onErrorContinue((err, i) -> {log.info("onErrorContinue={}", i);}) .reduce((i,j) -> i+j) .doOnNext(i -> System.out.println("sum=" + i)) .block(); }
秘訣是在 onErrorResume() 代碼塊的末尾添加 onErrorStop() ——這會阻塞 onErrorContinue(),這樣它就不會在 onErrorResume() 之前占用錯誤。嘗試刪除 onErrorStop(),你會看到 onErrorContinue() 在 onErrorResume 之前彈出。
到此這篇關于Reactor中的onErrorContinue 和 onErrorResume的文章就介紹到這了,更多相關Reactor onErrorContinue 內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Spring boot調用Oracle存儲過程的兩種方式及完整代碼
這篇文章主要給大家介紹了關于Spring boot調用Oracle存儲過程的兩種方式及完整代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧2020-08-08java基于正則表達式實現(xiàn)時間日期的常用判斷操作實例
這篇文章主要介紹了java基于正則表達式實現(xiàn)時間日期的常用判斷操作,簡單說明了正則表達式常用元字符含義并結合實例形式分析了java基于正則表達式針對常用日期時間格式的判斷操作技巧,需要的朋友可以參考下2017-10-10Java之next()、nextLine()區(qū)別及問題解決
這篇文章主要介紹了Java之next()、nextLine()區(qū)別及問題解決,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下2021-08-08