目錄
- 前言
- Spring Boot 版本
- 全局統(tǒng)一異常處理的前世今生
- Spring Boot的異常如何分類?
- 如何統(tǒng)一異常處理?
- 異常匹配的順序是什么?
- 總結(jié)
前言
軟件開發(fā)過程中難免遇到各種的BUG,各種的異常,一直就是在解決異常的路上永不停歇,如果你的代碼中再出現(xiàn)
try(){...}catch(){...}finally{...}
代碼塊,你還有心情看下去嗎?自己不覺得惡心嗎?冗余的代碼往往回喪失寫代碼的動(dòng)力,每天搬磚似的寫代碼,真的很難受。今天這篇文章教你如何去掉滿屏的
try(){...}catch(){...}finally{...}
,解放你的雙手。
Spring Boot 版本
本文基于的Spring Boot的版本是
2.3.4.RELEASE
。
全局統(tǒng)一異常處理的前世今生
早在
Spring 3.x
就已經(jīng)提出了
@ControllerAdvice
,可以與
@ExceptionHandler
、
@InitBinder
、
@ModelAttribute
等注解注解配套使用,這幾個(gè)此處就不再詳細(xì)解釋了。這幾個(gè)注解小眼一瞟只有
@ExceptionHandler
與異常有關(guān)啊,翻譯過來(lái)就是
異常處理器
。
其實(shí)異常的處理可以分為兩類,分別是局部異常處理
和全局異常處理
。
局部異常處理
:
@ExceptionHandler
和
@Controller
注解搭配使用,只有指定的controller層出現(xiàn)了異常才會(huì)被
@ExceptionHandler
捕獲到,實(shí)際生產(chǎn)中怕是有成百上千個(gè)controller了吧,顯然這種方式不合適。
全局異常處理
:既然局部異常處理不合適了,自然有人站出來(lái)解決問題了,于是就有了
@ControllerAdvice
這個(gè)注解的橫空出世了,
@ControllerAdvice
搭配
@ExceptionHandler
徹底解決了全局統(tǒng)一異常處理。當(dāng)然后面還出現(xiàn)了
@RestControllerAdvice
這個(gè)注解,其實(shí)就是
@ControllerAdvice
和
@ResponseBody
結(jié)晶。
Spring Boot的異常如何分類?
Java中的異常就很多,更別說(shuō)Spring Boot中的異常了,這里不再根據(jù)傳統(tǒng)意義上Java的異常進(jìn)行分類了,而是按照
controller
進(jìn)行分類,分為
進(jìn)入controller前的異常
和
業(yè)務(wù)層的異常
,如下圖:
進(jìn)入controller之前異常一般是
javax.servlet.ServletException
類型的異常,因此在全局異常處理的時(shí)候需要統(tǒng)一處理。幾個(gè)常見的異常如下:
NoHandlerFoundException
:客戶端的請(qǐng)求沒有找到對(duì)應(yīng)的controller,將會(huì)拋出404
異常。HttpRequestMethodNotSupportedException
:若匹配到了(匹配結(jié)果是一個(gè)列表,不同的是http方法不同,如:Get、Post等),則嘗試將請(qǐng)求的http方法與列表的控制器做匹配,若沒有對(duì)應(yīng)http方法的控制器,則拋該異常HttpMediaTypeNotSupportedException
:然后再對(duì)請(qǐng)求頭與控制器支持的做比較,比如content-type
請(qǐng)求頭,若控制器的參數(shù)簽名包含注解@RequestBody
,但是請(qǐng)求的content-type
請(qǐng)求頭的值沒有包含application/json
,那么會(huì)拋該異常(當(dāng)然,不止這種情況會(huì)拋這個(gè)異常)MissingPathVariableException
:未檢測(cè)到路徑參數(shù)。比如url為:/user/{userId},參數(shù)簽名包含@PathVariable("userId")
,當(dāng)請(qǐng)求的url為/user,在沒有明確定義url為/user的情況下,會(huì)被判定為:缺少路徑參數(shù)
如何統(tǒng)一異常處理?
在統(tǒng)一異常處理之前其實(shí)還有許多東西需要優(yōu)化的,比如統(tǒng)一結(jié)果返回的形式。當(dāng)然這里不再細(xì)說(shuō)了,不屬于本文范疇。
統(tǒng)一異常處理很簡(jiǎn)單,這里以前后端分離的項(xiàng)目為例,步驟如下:
- 新建一個(gè)統(tǒng)一異常處理的一個(gè)類
- 類上標(biāo)注
@RestControllerAdvice
這一個(gè)注解,或者同時(shí)標(biāo)注@ControllerAdvice
和@ResponseBody
這兩個(gè)注解。 - 在方法上標(biāo)注
@ExceptionHandler
注解,并且指定需要捕獲的異常,可以同時(shí)捕獲多個(gè)。
下面是作者隨便配置一個(gè)demo,如下:
/**
?*?全局統(tǒng)一的異常處理,簡(jiǎn)單的配置下,根據(jù)自己的業(yè)務(wù)要求詳細(xì)配置
?*/
@RestControllerAdvice
@Slf4j
public?class?GlobalExceptionHandler?{
????/**
?????*?重復(fù)請(qǐng)求的異常
?????*?@param?ex
?????*?@return
?????*/
????@ExceptionHandler(RepeatSubmitException.class)
????public?ResultResponse?onException(RepeatSubmitException?ex){
????????//打印日志
????????log.error(ex.getMessage());
????????//todo?日志入庫(kù)等等操作
????????//統(tǒng)一結(jié)果返回
????????return?new?ResultResponse(ResultCodeEnum.CODE_NOT_REPEAT_SUBMIT);
????}
????/**
?????*?自定義的業(yè)務(wù)上的異常
?????*/
????@ExceptionHandler(ServiceException.class)
????public?ResultResponse?onException(ServiceException?ex){
????????//打印日志
????????log.error(ex.getMessage());
????????//todo?日志入庫(kù)等等操作
????????//統(tǒng)一結(jié)果返回
????????return?new?ResultResponse(ResultCodeEnum.CODE_SERVICE_FAIL);
????}
????/**
?????*?捕獲一些進(jìn)入controller之前的異常,有些4xx的狀態(tài)碼統(tǒng)一設(shè)置為200
?????*?@param?ex
?????*?@return
?????*/
????@ExceptionHandler({HttpRequestMethodNotSupportedException.class,
????????????HttpMediaTypeNotSupportedException.class,?HttpMediaTypeNotAcceptableException.class,
????????????MissingPathVariableException.class,?MissingServletRequestParameterException.class,
????????????ServletRequestBindingException.class,?ConversionNotSupportedException.class,
????????????TypeMismatchException.class,?HttpMessageNotReadableException.class,
????????????HttpMessageNotWritableException.class,
????????????MissingServletRequestPartException.class,?BindException.class,
????????????NoHandlerFoundException.class,?AsyncRequestTimeoutException.class})
????public?ResultResponse?onException(Exception?ex){
????????//打印日志
????????log.error(ex.getMessage());
????????//todo?日志入庫(kù)等等操作
????????//統(tǒng)一結(jié)果返回
????????return?new?ResultResponse(ResultCodeEnum.CODE_FAIL);
????}
}
注意:
上面的只是一個(gè)例子,實(shí)際開發(fā)中還有許多的異常需要捕獲,比如TOKEN失效
、過期
等等異常,如果整合了其他的框架,還要注意這些框架拋出的異常,比如Shiro
,Spring Security
等等框架。異常匹配的順序是什么?
有些朋友可能疑惑了,如果我同時(shí)捕獲了父類和子類,那么到底能夠被那個(gè)異常處理器捕獲呢?比如
Exception
和
ServiceException
。此時(shí)可能就疑惑了,
這里先揭曉一下答案,當(dāng)然是ServiceException
的異常處理器捕獲了,精確匹配,如果沒有ServiceException
的異常處理器才會(huì)輪到它的父親
,父親
沒有才會(huì)到祖父
??傊痪湓?,精準(zhǔn)匹配,找那個(gè)關(guān)系最近的。為什么呢?這可不是憑空瞎說(shuō)的,源碼為證,出處
org.springframework.web.method.annotation.ExceptionHandlerMethodResolver#getMappedMethod
,如下:
@Nullable
?private?Method?getMappedMethod(Class?extends?Throwable>?exceptionType)?{
??List>?matches?=?new?ArrayList<>();
????//遍歷異常處理器中定義的異常類型
??for?(Class?extends?Throwable>?mappedException?:?this.mappedMethods.keySet())?{
??????//是否是拋出異常的父類,如果是添加到集合中
???if?(mappedException.isAssignableFrom(exceptionType))?{????
????????//添加到集合中
????matches.add(mappedException);??
???}
??}
????//如果集合不為空,則按照規(guī)則進(jìn)行排序
??if?(!matches.isEmpty())?{
???matches.sort(new?ExceptionDepthComparator(exceptionType));
??????//取第一個(gè)
???return?this.mappedMethods.get(matches.get(0));
??}
??else?{
???return?null;
??}
?}
在初次異常處理的時(shí)候會(huì)執(zhí)行上述的代碼找到最匹配的那個(gè)異常處理器方法,后續(xù)都是直接從緩存中(一個(gè)Map
結(jié)構(gòu),key
是異常類型,value
是異常處理器方法)。別著急,上面代碼最精華的地方就是對(duì)
matches
進(jìn)行排序的代碼了,我們來(lái)看看
ExceptionDepthComparator
這個(gè)比較器的關(guān)鍵代碼,如下:
//遞歸調(diào)用,獲取深度,depth值越小越精準(zhǔn)匹配
private?int?getDepth(Class>?declaredException,?Class>?exceptionToMatch,?int?depth)?{
????//如果匹配了,返回
??if?(exceptionToMatch.equals(declaredException))?{
???//?Found?it!
???return?depth;
??}
??//?遞歸結(jié)束的條件,最大限度了
??if?(exceptionToMatch?==?Throwable.class)?{
???return?Integer.MAX_VALUE;
??}
????//繼續(xù)匹配父類
??return?getDepth(declaredException,?exceptionToMatch.getSuperclass(),?depth? ?1);
?}
精髓全在這里了,一個(gè)遞歸搞定,計(jì)算深度,depth
初始值為0。值越小,匹配度越高越精準(zhǔn)。總結(jié)
全局異常的文章萬(wàn)萬(wàn)千,能夠講清楚的能有幾篇呢?
只出最精的文章,做最野的程序員,如果覺得不錯(cuò)的,關(guān)注分享走一波,謝謝支持?。?!