SpringMVC使用@ExceptionHandler注解在Controller中處理異常
異常是每一個(gè)應(yīng)用必須要處理的問(wèn)題
Spring MVC項(xiàng)目,如果不做任何的異常處理的話,發(fā)生異常后,異常堆棧信息會(huì)直接拋出到頁(yè)面。
在Controller寫(xiě)一個(gè)異常
@GetMapping(value="/hello",produces={"text/html; charset=UTF-8"})
@ResponseBody
public String hello(ModelAndView model){
int c = 100 / 0;
return "<h1>User info</h1>";
}瀏覽器訪問(wèn):

用戶體驗(yàn)相當(dāng)不好,所以一般情況下,應(yīng)用必須要想辦法處理異常。
異常處理方式
常見(jiàn)的異常處理方式,無(wú)非:
- 應(yīng)用中對(duì)所有可能發(fā)生異常的地方,都try catch,捕獲異常后做相應(yīng)的處理。
- 集中處理異常。
第一種方式顯然不好,一方面是代碼中需要到處都寫(xiě)try catch,萬(wàn)一某一段代碼由于程序員的疏忽沒(méi)有寫(xiě),異常就會(huì)拋出到前臺(tái),很不好。
另外,某些情況下異常是不能捕獲的,比如需要事務(wù)處理的代碼,捕獲異常后會(huì)影響到事務(wù)回滾。
所以,我們還是需要想辦法用第2中方式來(lái)處理異常。n多年前曾經(jīng)做過(guò)一個(gè)摩托羅拉的項(xiàng)目,其中一項(xiàng)需求就是對(duì)一個(gè)線上系統(tǒng)的異常做處理、不允許異常信息拋出到前臺(tái)頁(yè)面。當(dāng)初那個(gè)項(xiàng)目并沒(méi)有采用類(lèi)似SpringMVC的框架,所以最終提出了用filter、在filter中攔截異常的處理方案并且被客戶采納。
其實(shí)SpringMVC的統(tǒng)一異常處理方案和我上面項(xiàng)目中采用的方案非常類(lèi)似。
SpringMVC的異常處理
Spring MVC提供了如下異常處理選項(xiàng):
- @ExceptionHandle
- @ExceptionHandle + @ControllerAdvice
- 自定義異常處理器HandlerExceptionResolver
@ExceptionHandle
@ExceptionHandle可以作用在Controller中,比如,在上面發(fā)生異常的Controller中增加一個(gè)@ExceptionHandle注解的方法:
@GetMapping(value="/hello",produces={"text/html; charset=UTF-8"})
@ResponseBody
public String hello(ModelAndView model){
int c = 100 / 0;
return "<h1>User info</h1>";
}
@ExceptionHandler(Exception.class)
@ResponseBody
public String handle(Exception ex){
return "錯(cuò)了在helloworld Controller error msg is ===";
}運(yùn)行:

說(shuō)明Hello方法中發(fā)生的異常,已經(jīng)被handle方法處理,前臺(tái)頁(yè)面不再會(huì)出現(xiàn)異常信息。
問(wèn)題解決了!
但是@ExceptionHandle只在當(dāng)前Controller文件中生效,也就是說(shuō),當(dāng)前Controller中的方法、或者方法調(diào)用的service層、dao層等發(fā)生的異常,才會(huì)被捕獲到。其他Controller中發(fā)生的異常依然不會(huì)被捕獲。
這樣的話,就需要對(duì)每一個(gè)Controller增加@ExceptionHandle進(jìn)行處理,處理起來(lái)還是有點(diǎn)麻煩。
統(tǒng)一異常處理
@ExceptionHandle + @ControllerAdvice可以實(shí)現(xiàn)統(tǒng)一異常處理。
@ControllerAdvice注解可以實(shí)現(xiàn):
On startup, RequestMappingHandlerMapping and ExceptionHandlerExceptionResolver detect controller advice beans and apply them at runtime. Global @ExceptionHandler methods, from an @ControllerAdvice, are applied after local ones, from the @Controller. By contrast, global @ModelAttribute and @InitBinder methods are applied before local ones.
SpringMVC啟動(dòng)的過(guò)程中,RequestMappingHandlerMapping和ExceptionHandlerExceptionResolver會(huì)檢測(cè)到advice bean并且在運(yùn)行時(shí)會(huì)使用他們。在@ControllerAdvice中的@ExceptionHandler會(huì)變成一個(gè)全局的異常處理器、在本地異常處理器之后生效。并且,@ModelAttribute和 @InitBinder會(huì)在本地的之后生效。
這段話的意思就是,@ExceptionHandle + @ControllerAdvice之后,@ExceptionHandle就會(huì)變成全局異常處理器。所謂的本地異常處理器,就是寫(xiě)在Controller中的@ExceptionHandle異常處理器。
這個(gè)工作是ExceptionHandlerExceptionResolver干的,源碼中可以看到:
@Nullable
protected ServletInvocableHandlerMethod getExceptionHandlerMethod(
@Nullable HandlerMethod handlerMethod, Exception exception) {
Class<?> handlerType = null;
//先找"local"異常處理器
if (handlerMethod != null) {
// Local exception handler methods on the controller class itself.
// To be invoked through the proxy, even in case of an interface-based proxy.
handlerType = handlerMethod.getBeanType();
ExceptionHandlerMethodResolver resolver = this.exceptionHandlerCache.get(handlerType);
if (resolver == null) {
resolver = new ExceptionHandlerMethodResolver(handlerType);
this.exceptionHandlerCache.put(handlerType, resolver);
}
Method method = resolver.resolveMethod(exception);
if (method != null) {
return new ServletInvocableHandlerMethod(handlerMethod.getBean(), method);
}
// For advice applicability check below (involving base packages, assignable types
// and annotation presence), use target class instead of interface-based proxy.
if (Proxy.isProxyClass(handlerType)) {
handlerType = AopUtils.getTargetClass(handlerMethod.getBean());
}
}
//再找advice的異常處理器
for (Map.Entry<ControllerAdviceBean, ExceptionHandlerMethodResolver> entry : this.exceptionHandlerAdviceCache.entrySet()) {
ControllerAdviceBean advice = entry.getKey();
if (advice.isApplicableToBeanType(handlerType)) {
ExceptionHandlerMethodResolver resolver = entry.getValue();
Method method = resolver.resolveMethod(exception);
if (method != null) {
return new ServletInvocableHandlerMethod(advice.resolveBean(), method);
}
}
}
return null;
}驗(yàn)證一下。
加一個(gè)MyGlobalExceptionController:
@ControllerAdvice
public class MyGlobalExceptionController {
@ExceptionHandler(Exception.class)
@ResponseBody
public String handle(Exception ex){
return return "錯(cuò)了MyGlobalExceptionController error msg is ===";
}
}先不去掉HelloWorldController中的異常處理器,運(yùn)行hello:

生效的是HelloWorldController中的異常處理器。
然后去掉HelloWorldController中的異常處理器:

寫(xiě)在MyGlobalExceptionController中的全局異常處理器生效!
自定義異常處理器HandlerExceptionResolver
實(shí)現(xiàn)接口HandlerExceptionResolver,或者擴(kuò)展虛擬類(lèi)AbstractHandlerMethodExceptionResolver(勉強(qiáng)可以考慮),定義自己的異常處理器。
其實(shí),非必要(沒(méi)想到有什么必要性)不建議!
以上就是SpringMVC使用@ExceptionHandler注解在Controller中處理異常的詳細(xì)內(nèi)容,更多關(guān)于SpringMVC異常處理的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Spring中的@ExceptionHandler注解統(tǒng)一異常處理詳解
- Spring的異常處理@ExceptionHandler注解解析
- 關(guān)于SpringBoot使用@ExceptionHandler注解局部異常處理
- Spring中@ExceptionHandler注解的使用方式
- Spring中@ExceptionHandler注解的工作原理詳解
- Spring @ExceptionHandler注解統(tǒng)一異常處理和獲取方法名
- Spring中的@ControllerAdvice和@ExceptionHandler注解處理全局異常
- Spring中的@ExceptionHandler注解詳解與應(yīng)用示例
相關(guān)文章
java時(shí)間戳轉(zhuǎn)換為日期格式的多種方式
本文介紹了五種將Java時(shí)間戳轉(zhuǎn)換為日期格式的方法,包括使用Date類(lèi)、LocalDateTime類(lèi)、Instant類(lèi)、DateUtils類(lèi)以及自定義時(shí)區(qū),每種方法都有其適用場(chǎng)景,可以根據(jù)具體需求選擇合適的方法,感興趣的朋友跟隨小編一起看看吧2025-01-01
Java實(shí)現(xiàn)導(dǎo)出Excel功能
通過(guò)java中Controller層,來(lái)接受請(qǐng)求,數(shù)據(jù)庫(kù)查詢(xún)到的數(shù)據(jù)進(jìn)行封裝,然后使用ExcelUtils進(jìn)行輸出,接下來(lái)通過(guò)本文給大家分享Java實(shí)現(xiàn)導(dǎo)出Excel功能的實(shí)例代碼,感興趣的朋友跟隨小編一起看看吧2021-11-11
Java多線程Callable接口實(shí)現(xiàn)代碼示例
相信大家對(duì)Java編程中如何創(chuàng)建線程已經(jīng)不陌生了,這篇文章就向朋友們介紹實(shí)現(xiàn)callable接口,具體實(shí)例詳見(jiàn)正文。2017-10-10
Windows編寫(xiě)jar啟動(dòng)腳本和關(guān)閉腳本的操作方法
腳本文件,通常放入/bin目錄下,編寫(xiě)啟動(dòng)腳本需要保證能夠識(shí)別到對(duì)應(yīng)的jar文件,其次需要保證能夠識(shí)別到/config中的配置文件信息,這篇文章主要介紹了Windows編寫(xiě)jar啟動(dòng)腳本和關(guān)閉腳本的操作方法,需要的朋友可以參考下2022-12-12
mybatis查詢(xún)實(shí)現(xiàn)返回List<Map>類(lèi)型數(shù)據(jù)操作
這篇文章主要介紹了mybatis查詢(xún)實(shí)現(xiàn)返回List<Map>類(lèi)型數(shù)據(jù)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-11-11

