SpringMVC 三種異常處理方式
SpringMVC 三種異常處理方式
在 SpringMVC, SpringBoot 處理 web 請求時, 若遇到錯誤或者異常,返回給使用者一個良好的錯誤資訊比 Whitelabel Error Page 好的多。 SpringMVC 提供了三種異常處理方式, 良好的運用它們可以給使用者提供可讀的錯誤資訊。
1. 實現 HandlerExceptionResolver
public class AppHandlerExceptionResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { ModelAndView mav = new ModelAndView(); mav.addObject("message", ex.getMessage()); // 可以設定檢視名導向錯誤頁面 mav.setViewName("/error"); // 直接返回檢視 // 如果返回 null,則會呼叫下一個 HandlerExceptionResolver return mav; } }
然後配置一個 HandlerExceptionResolver
@Bean
public AppHandlerExceptionResolver appHandlerExceptionResolver() {
return new AppHandlerExceptionResolver();
}
HandlerExceptionResolver 的實現類會 catch 到 @Controller 方法執行時發生的異常,處理後返回 ModelAndView 作為結果檢視,因此可以通過它來定製異常檢視。
HandlerExceptionResolver 只能捕獲 @Controller 層發生的異常(包括 @Controller 呼叫 @Service 發生的異常),其他地方的異常,比如訪問了一個不存在的路徑,不會被 HandlerExceptionResolver 捕獲,此時會跳到 ErrorController 處理, 下面會說到。
2. 通過 @ControllerAdvice 和 @ExceptionHandler 註解
// 可以配置攔截指定的類或者包等 // @RestControllerAdvice 使 @ExceptionHandler 註解的方法預設具有 @ResponseBody 註解 @RestControllerAdvice(basePackageClasses = HelloWorldController.class) public class AppExceptionHandlerAdvice { // 配置攔截的錯誤型別 // 這裡也可以返回 ModelAndView 導向錯誤檢視 @ExceptionHandler(Exception.class) public ResponseEntity<Object> responseEntity(Exception e) { HttpHeaders headers = new HttpHeaders(); headers.setContentType(MediaType.APPLICATION_JSON_UTF8); Map<String, Object> map = new HashMap<>(); map.put("status", 400); map.put("message", e.getMessage()); // 直接返回結果 return new ResponseEntity<>(map, headers, HttpStatus.BAD_REQUEST); } }
這種方式配置的異常處理由 HandlerExceptionResolver 的預設實現類 HandlerExceptionResolverComposite 處理,因此也只能捕獲 @Controller 層的異常。
@ExceptionHandler 可以返回 ModelAndView 定製異常檢視。
@ControllerAdvice 可以攔截特定的類,@ExceptionHandler 可以攔截特定的異常,因此可以更精確的配置異常處理邏輯。
3. 自定義 ErrorController bean
@RestController
@RequestMapping("/error")
public class AppErrorController extends AbstractErrorController {
public AppErrorController(ErrorAttributes errorAttributes) {
super(errorAttributes);
}
@RequestMapping
public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) {
Map<String, Object> body = getErrorAttributes(request, false);
HttpStatus status = getStatus(request);
// 返回響應體
return new ResponseEntity<>(body, status);
}
@Override
public String getErrorPath() {
return "/error";
}
}
如果沒有配置 ErrorController, SpringBoot 會通過 ErrorMvcAutoConfiguration 自動配置一個,預設的實現類為 BasicErrorController。
ErrorController 可以處理非 @Controller 層丟擲的異常,例如常見的訪問了一個不存在的路徑。
ErrorController 可以進行統一的錯誤處理,即讓 HandlerExceptionResolver 返回的 ModelAndView 導向錯誤頁面