詳解SpringBoot異常處理流程及原理

異常處理流程

執行目標方法,目標方法運行期間有任何異常都會被catch捕獲,並標志當前請求結束,dispatchException拋出異常

在這裡插入圖片描述

進入視圖解析流程,並渲染頁面,發生異常時,參數mv為空,傳入捕獲的異常dispatchException

在這裡插入圖片描述

處理handler發生的異常,處理完成返回ModelAndView

在這裡插入圖片描述

(1)遍歷所有的HandlerExceptionResolvers,找到可以處理當前異常的解析器來解析異常

在這裡插入圖片描述

(2)調用resolveException解析異常,傳入requestresponse對象,哪個方法,發生的異常,然後自定義異常處理返回ModelAndView

在這裡插入圖片描述

(3)系統默認的異常解析器

在這裡插入圖片描述

DefaultErrorAttributes先來處理異常,把異常信息保存到request域並返回null

在這裡插入圖片描述

ExceptionHandlerExceptionResolver用來處理標註瞭@ExceptionHandler註解的方法異常

ResponseStatusExceptionResolver用來處理標註瞭@ResponseStatus註解的方法異常

DefaultHandlerExceptionResolver默認的處理器異常解析器,處理一些常見的異常

(4)如果沒有任何解析器能夠處理異常,異常就會拋出

在這裡插入圖片描述

(5)如果沒有任何解析器能夠處理當前異常,最終就會發送/error請求,將保存的異常信息轉發到/errorBasicErrorController專門來處理/error請求,BasicErrorController會遍歷所有的ErrorViewResolver解析錯誤視圖,如果沒有自定義的錯誤視圖解析器,就會使用默認的DefaultErrorViewResolver,會把響應碼作為錯誤頁的地址,模板引擎最終響應這個頁面。

幾種異常處理方式及原理

1.自定義錯誤頁,error/404.htmlerror/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請求就會默認轉給basicErrorControllerBasicErrorController專門來處理/error請求,適配4xx.html或者5xx.html頁面

(2)如果異常沒有任何解析器能處理,tomcat底層 也會調用response.sendErrorerror請求就會默認轉給basicErrorControllerBasicErrorController專門來處理/error請求,適配4xx.html或者5xx.html頁面。

(3)basicErrorController 要去的頁面地址是由 ErrorViewResolver這個錯誤視圖解析器決定的,即適配4xx.html或者5xx.html頁面。

到此這篇關於詳解SpringBoot異常處理流程及原理的文章就介紹到這瞭,更多相關SpringBoot異常處理流程及原理內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: