當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > 架構(gòu)師社區(qū)
[導(dǎo)讀]目錄前言SpringBoot版本全局統(tǒng)一異常處理的前世今生SpringBoot的異常如何分類?如何統(tǒng)一異常處理?異常匹配的順序是什么?總結(jié)前言軟件開發(fā)過程中難免遇到各種的BUG,各種的異常,一直就是在解決異常的路上永不停歇,如果你的代碼中再出現(xiàn)try(){...}catch(){...

目錄

  • 前言
  • 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ù)層的異常,如下圖:

滿屏的try-catch,你不瘆得慌?
進(jìn)入controller之前異常一般是javax.servlet.ServletException類型的異常,因此在全局異常處理的時(shí)候需要統(tǒng)一處理。幾個(gè)常見的異常如下:

  1. NoHandlerFoundException:客戶端的請(qǐng)求沒有找到對(duì)應(yīng)的controller,將會(huì)拋出404異常。
  2. HttpRequestMethodNotSupportedException:若匹配到了(匹配結(jié)果是一個(gè)列表,不同的是http方法不同,如:Get、Post等),則嘗試將請(qǐng)求的http方法與列表的控制器做匹配,若沒有對(duì)應(yīng)http方法的控制器,則拋該異常
  3. HttpMediaTypeNotSupportedException:然后再對(duì)請(qǐng)求頭與控制器支持的做比較,比如content-type請(qǐng)求頭,若控制器的參數(shù)簽名包含注解@RequestBody,但是請(qǐng)求的content-type請(qǐng)求頭的值沒有包含application/json,那么會(huì)拋該異常(當(dāng)然,不止這種情況會(huì)拋這個(gè)異常)
  4. 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)目為例,步驟如下

  1. 新建一個(gè)統(tǒng)一異常處理的一個(gè)類
  2. 類上標(biāo)注@RestControllerAdvice這一個(gè)注解,或者同時(shí)標(biāo)注@ControllerAdvice@ResponseBody這兩個(gè)注解。
  3. 在方法上標(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è)異常處理器捕獲呢?比如ExceptionServiceException。

此時(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?exceptionType)?{
??List>?matches?=?new?ArrayList<>();
????//遍歷異常處理器中定義的異常類型
??for?(Class?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)注分享走一波,謝謝支持?。?!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉