SpringBoot错误处理逻辑
刘婉晴 人气:01. 自定义错误页面
将自定义错误页面放在 templates 的 error 文件夹下,SpringBoot 精确匹配错误信息,使用 4xx.html 或者 5xx.html 页面可以打印错误信息
4xx —— 打印 status 及 message 信息
<h2 th:text="${status}">page not found</h2> <h3 th:text="${#message}">We Couldn't Find This Page</h3>
5xx—— 打印 message 及 trace 信息
<h3 th:text="${message}">Something went wrong.</h3> <p class="nrml-txt" th:text="${trace}">Why not try refreshing you page? Or you can <a href="#" rel="external nofollow" >contact our support</a> if the problem persists.</p>
2. @ControllerAdvice+@ExceptionHandler
自定义全局异常处理类,处理 ArithmeticException 及 NullPointerException 异常
package com.wanqing.admin.exception; import lombok.extern.slf4j.Slf4j; import org.springframework.web.bind.annotation.ControllerAdvice; import org.springframework.web.bind.annotation.ExceptionHandler; @Slf4j @ControllerAdvice // 使用此注释 public class GlobalExceptionHandler { @ExceptionHandler({ArithmeticException.class, NullPointerException.class }) // 使用此注释,大括号内为可以处理的异常信息 public String handleArithException(Exception e){ log.info("异常是:" + e); return "login"; // 返回一个视图地址(ModelAndView) } }
原理:
使用 ExceptionHandlerExceptionResolver 异常处理器处理用 @ExceptionHandler 注释的异常
3. 使用@ResponseStatus处理自定义异常
自定义异常类示例代码:
import org.springframework.http.HttpStatus; import org.springframework.web.bind.annotation.ResponseStatus; @ResponseStatus(value = HttpStatus.FORBIDDEN, reason = "用户数量太多~~") // 异常可以返回状态码信息 public class userToMany extends RuntimeException{ // 有参构造器 public userToMany(String message){ super(message); } public userToMany(){ } }
原理:
ResponseStatusExceptionResolver 处理器可以处理 @ResponseStatus 注解的异常,得到 @ResponseStatus 注解的信息,调用 response.sendError(statusCode) 方法将错误信息返回,并发送 /error 请求,交由底层处理
sendError —— 表示此次请求立刻结束,发出 /error 请求,SpringBoot 找谁能处理,都不能处理返回默认的错误页
protected ModelAndView applyStatusAndReason(int statusCode, @Nullable String reason, HttpServletResponse response) throws IOException { if (!StringUtils.hasLength(reason)) { response.sendError(statusCode); } else { String resolvedReason = this.messageSource != null ? this.messageSource.getMessage(reason, (Object[])null, reason, LocaleContextHolder.getLocale()) : reason; response.sendError(statusCode, resolvedReason); } return new ModelAndView(); }
4. 框架底层异常
使用 DefaultHandlerExceptionResolver 异常处理器能处理 SpringMVC 底层异常,其能处理我异常种类如下
protected ModelAndView doResolveException(HttpServletRequest request, HttpServletResponse response, @Nullable Object handler, Exception ex) { try { if (ex instanceof HttpRequestMethodNotSupportedException) { return this.handleHttpRequestMethodNotSupported((HttpRequestMethodNotSupportedException)ex, request, response, handler); } // 此处省略............. if (ex instanceof HttpMessageNotWritableException) { return this.handleHttpMessageNotWritable((HttpMessageNotWritableException)ex, request, response, handler); } if (ex instanceof MethodArgumentNotValidException) { return this.handleMethodArgumentNotValidException((MethodArgumentNotValidException)ex, request, response, handler); } } catch (Exception var6) { if (this.logger.isWarnEnabled()) { this.logger.warn("Failure while trying to resolve exception [" + ex.getClass().getName() + "]", var6); } } return null; }
5. 自定义异常解析器
自定义异常解析器需要满足以下:
- 实现 HandlerExceptionResolver 接口并注册到容器中(@Component)
- 在自定义解析器中实现 resolveException 方法,方法内可通过 sendError 方法返回错误信息并返回一空视图,交给底层将错误信息解析拼接为最终页面
- 可以通过 @Order 注释调整自定义异常解析器的优先级,value越小优先级越高
自定义异常解析器示例代码:
import javax.servlet.http.HttpServletResponse; import java.io.IOException; @Order(value = Ordered.HIGHEST_PRECEDENCE) // 优先级数字越小,优先级越高 @Component // 注册到容器中 public class CustomerHandlerResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) { try { httpServletResponse.sendError(511, "我不喜欢的错误"); } catch (IOException e1){ e1.printStackTrace(); } return new ModelAndView(); } }
自定义异常处理器被加入(未调整优先级时,默认加到最后):
6. ErrorViewResolver实现自定义处理异常
交由ErrorViewResolver的情况 :
情况一: response.sendError ,error 请求转给 Controller
情况二: 判断异常请求无人处理,tomcat 底层 response.sendError
ErrorViewResolver处理方法:
BasicErrorController 得到要去的地址后由 ErrorViewResolver 解析,解析规则为拼接状态码等信息
—— 即 ErrorViewResolver 是最后的异常处理, 没人处理的异常,被它处理
加载全部内容