Spring boot項目中異常攔截設(shè)計和處理詳解
背景:
項目運行過程中會出現(xiàn)各種各樣的問題,常見的有以下幾種情況:
- 業(yè)務(wù)流程分析疏漏,對業(yè)務(wù)流程的反向操作、邊界分析設(shè)計不充分
- 調(diào)用外部服務(wù)、調(diào)用外部系統(tǒng)出現(xiàn)的超時、錯誤、返回值與預(yù)期不符
- 外部資源連通性問題,db等服務(wù)器出現(xiàn)的網(wǎng)絡(luò)抖動或宕機
無論是分析設(shè)計、開發(fā)、測試、線上都需要能夠準確定位問題并制定解決方案。
目的:
- 規(guī)范化異常的處理過程,避免異常被吞和到處都在捕獲異常的情況
- 準確的反饋異常信息,為定位問題提供依據(jù)
- 通用性異常全局處理,降低業(yè)務(wù)開發(fā)關(guān)注度
- 對異常情況進行預(yù)警,以便能夠及時響應(yīng)
一、異常規(guī)劃
1. 業(yè)務(wù)類異常
造成業(yè)務(wù)流程不能正確執(zhí)行的行為,常見的幾種:
- 輸入必填驗證
- 業(yè)務(wù)狀態(tài)約束校驗
- 權(quán)限驗證
- 調(diào)用外部服務(wù)返回數(shù)據(jù)不符合預(yù)期
這類異常需要給調(diào)用方返回明確的異常描述信息,一般情況下和代碼無關(guān),無需調(diào)整編碼
注:是業(yè)務(wù)完整性的一部分,需提前分析
2. 系統(tǒng)類異常
服務(wù)調(diào)用異常: 超時、中斷、接口異常(非200請求)
第三方異常 :db\redis\消息隊列 連接失敗等
注:通常與業(yè)務(wù)流程無關(guān),與第三方系統(tǒng)有關(guān),不能簡單的通過調(diào)整代碼解決
3. 通用異常
編碼不嚴謹、數(shù)據(jù)異常造成的問題,不可預(yù)測
舉例:參數(shù)類型不匹配、空指針、數(shù)組越界
二、異常攔截
在springboot中全局異常攔截處理已知的有下面2種方案:
方案1:@ControllerAdvice、實現(xiàn)ErrorController
注:利用springboot自帶的攔截機制,只需要定義出處理的策略,沒有破壞springboot的約定
方案2:繼承AbstractHandlerExceptionResolver,完全自定義處理策略
注:使用spring中最底層的類,打破了springboot的約定,能夠攔截到所有異常
三、方案實踐
筆者基于方案一進行實踐。
1. 異常攔截時序圖
2. RrcRestAdvice實現(xiàn)代碼

2. RrcExpHandler實現(xiàn)代碼

注意:基于RestControllerAdvice的異常攔截只能捕獲請求達controller之后的程序異常,所以需要實現(xiàn)ErrorController處理之前的異常。
總結(jié):
推薦基于springboot中@ControllerAdvice 和 ErrorController接口的約定,相對較符合springboot的約定。
其他可選方案:
繼承AbstractHandlerExceptionResolver
優(yōu)點:可完全自定義處理策略。缺點:對框架約定破壞較為嚴重,自定義處理策略容易疏漏。
繼承HandlerInterceptorAdapter
理論上可以處理業(yè)務(wù)代碼拋出的異常,優(yōu)缺點沒有進行過驗證。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
SpringBoot測試時卡在Resolving Maven dependencies的問題
這篇文章主要介紹了SpringBoot測試時卡在Resolving Maven dependencies的問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02
java 中JDBC連接數(shù)據(jù)庫代碼和步驟詳解及實例代碼
這篇文章主要介紹了java 中JDBC連接數(shù)據(jù)庫代碼和步驟詳解及實例代碼的相關(guān)資料,需要的朋友可以參考下2017-02-02
Spring MVC請求參數(shù)與響應(yīng)結(jié)果全局加密和解密詳解
這篇文章主要給大家介紹了關(guān)于Spring MVC請求參數(shù)與響應(yīng)結(jié)果全局加密和解密的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-08-08
springboot如何讀取配置文件(application.yml)中的屬性值
本篇文章主要介紹了springboot如何讀取配置文件(application.yml)中的屬性值,具有一定的參考價值,有興趣的小伙伴可以了解一下2017-04-04

