詳解SpringBoot異常處理流程及原理
異常處理流程
執行目標方法,目標方法運行期間有任何異常都會被catch
捕獲,並標志當前請求結束,dispatchException
拋出異常
進入視圖解析流程,並渲染頁面,發生異常時,參數mv
為空,傳入捕獲的異常dispatchException
處理handler
發生的異常,處理完成返回ModelAndView
(1)遍歷所有的HandlerExceptionResolvers
,找到可以處理當前異常的解析器來解析異常
(2)調用resolveException
解析異常,傳入request
和response
對象,哪個方法,發生的異常,然後自定義異常處理返回ModelAndView
(3)系統默認的異常解析器
① DefaultErrorAttributes
先來處理異常,把異常信息保存到request
域並返回null
② ExceptionHandlerExceptionResolver
用來處理標註瞭@ExceptionHandler
註解的方法異常
③ ResponseStatusExceptionResolver
用來處理標註瞭@ResponseStatus
註解的方法異常
④ DefaultHandlerExceptionResolver
默認的處理器異常解析器,處理一些常見的異常
(4)如果沒有任何解析器能夠處理異常,異常就會拋出
(5)如果沒有任何解析器能夠處理當前異常,最終就會發送/error
請求,將保存的異常信息轉發到/error
。BasicErrorController
專門來處理/error
請求,BasicErrorController
會遍歷所有的ErrorViewResolver
解析錯誤視圖,如果沒有自定義的錯誤視圖解析器,就會使用默認的DefaultErrorViewResolver
,會把響應碼作為錯誤頁的地址,模板引擎最終響應這個頁面。
幾種異常處理方式及原理
1.自定義錯誤頁,error/404.html
、error/5xx.html
。有精確的錯誤狀態碼頁面就匹配精確,沒有就找 4xx.html
,如果都沒有就觸發白頁
2.使用@ControllerAdvice
和@ExceptionHandler
處理全局異常,底層是ExceptionHandlerExceptionResolver
支持的
3.使用@ResponseStatus
和自定義異常。底層是 ResponseStatusExceptionResolver
,底層調用 response.sendError(statusCode, resolvedReason)
,Tomcat會收到一個error
。請求最後new
一個空的ModelAndView
返回,這樣任何處理解析器都處理不瞭當前的異常,最終就會發送/error
請求,BasicErrorController
專門來處理/error
請求,適配4xx.html
或者5xx.html
頁面
4.Spring底層的異常,如參數類型轉換異常。底層是DefaultHandlerExceptionResolver
處理框架底層的異常,底層也是response.sendError(HttpServletResponse.SC_NOT_ACCEPTABLE)
,Tomcat會收到一個error
。請求最後new
一個空的ModelAndView
返回,這樣任何處理解析器都處理不瞭當前的異常,最終就會發送/error
請求,BasicErrorController
專門來處理/error
請求,適配4xx.html
或者5xx.html
頁面
protected ModelAndView doResolveException( HttpServletRequest request, HttpServletResponse response, @Nullable Object handler, Exception ex) { try { if (ex instanceof HttpRequestMethodNotSupportedException) { return handleHttpRequestMethodNotSupported( (HttpRequestMethodNotSupportedException) ex, request, response, handler); } else if (ex instanceof HttpMediaTypeNotSupportedException) { return handleHttpMediaTypeNotSupported( (HttpMediaTypeNotSupportedException) ex, request, response, handler); } else if (ex instanceof HttpMediaTypeNotAcceptableException) { return handleHttpMediaTypeNotAcceptable( (HttpMediaTypeNotAcceptableException) ex, request, response, handler); } else if (ex instanceof MissingPathVariableException) { return handleMissingPathVariable( (MissingPathVariableException) ex, request, response, handler); } else if (ex instanceof MissingServletRequestParameterException) { return handleMissingServletRequestParameter( (MissingServletRequestParameterException) ex, request, response, handler); } else if (ex instanceof ServletRequestBindingException) { return handleServletRequestBindingException( (ServletRequestBindingException) ex, request, response, handler); } else if (ex instanceof ConversionNotSupportedException) { return handleConversionNotSupported( (ConversionNotSupportedException) ex, request, response, handler); } else if (ex instanceof TypeMismatchException) { return handleTypeMismatch( (TypeMismatchException) ex, request, response, handler); } else if (ex instanceof HttpMessageNotReadableException) { return handleHttpMessageNotReadable( (HttpMessageNotReadableException) ex, request, response, handler); } else if (ex instanceof HttpMessageNotWritableException) { return handleHttpMessageNotWritable( (HttpMessageNotWritableException) ex, request, response, handler); } else if (ex instanceof MethodArgumentNotValidException) { return handleMethodArgumentNotValidException( (MethodArgumentNotValidException) ex, request, response, handler); } else if (ex instanceof MissingServletRequestPartException) { return handleMissingServletRequestPartException( (MissingServletRequestPartException) ex, request, response, handler); } else if (ex instanceof BindException) { return handleBindException((BindException) ex, request, response, handler); } else if (ex instanceof NoHandlerFoundException) { return handleNoHandlerFoundException( (NoHandlerFoundException) ex, request, response, handler); } else if (ex instanceof AsyncRequestTimeoutException) { return handleAsyncRequestTimeoutException( (AsyncRequestTimeoutException) ex, request, response, handler); } } catch (Exception handlerEx) { if (logger.isWarnEnabled()) { logger.warn("Failure while trying to resolve exception [" + ex.getClass().getName() + "]", handlerEx); } } return null; }
5.自定義實現 HandlerExceptionResolver
處理異常,可以作為默認的全局異常處理規則
@Order(value = Ordered.HIGHEST_PRECEDENCE) @Component public class CustomerHandlerExceptionResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { try { response.sendError(521,"I love you !"); } catch (IOException e) { e.printStackTrace(); } return new ModelAndView(); } }
ErrorViewResolver
實現自定義處理異常。
(1)底層調用response.sendError
時 ,error
請求就會默認轉給basicErrorController
,BasicErrorController
專門來處理/error
請求,適配4xx.html
或者5xx.html
頁面
(2)如果異常沒有任何解析器能處理,tomcat底層 也會調用response.sendError
。error
請求就會默認轉給basicErrorController
,BasicErrorController
專門來處理/error
請求,適配4xx.html
或者5xx.html
頁面。
(3)basicErrorController
要去的頁面地址是由 ErrorViewResolver
這個錯誤視圖解析器決定的,即適配4xx.html
或者5xx.html
頁面。
到此這篇關於詳解SpringBoot異常處理流程及原理的文章就介紹到這瞭,更多相關SpringBoot異常處理流程及原理內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 詳解SpringMVC中的異常處理
- SpringBoot的異常處理流程是什麼樣的?
- SpringMVC全局異常處理的三種方式
- SpringBoot錯誤處理流程深入詳解
- springboot過濾器和攔截器的實例代碼