SpringBoot如何自定义错误处理逻辑

2023-06-19 09:28:44 逻辑 错误 自定义

本篇内容主要讲解“SpringBoot如何自定义错误处理逻辑”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“SpringBoot如何自定义错误处理逻辑”吧!

1. 自定义错误页面

将自定义错误页面放在 templates 的 error 文件夹下,SpringBoot 精确匹配错误信息,使用 4xx.html 或者 5xx.html 页面可以打印错误信息

4xx —— 打印 status 及 message 信息

  <h3 th:text="${status}">page not found</h3>
  <h4 th:text="${#message}">We Couldn't Find This Page</h4>

5xx&mdash;&mdash; 打印 message 及 trace 信息

	<h4 th:text="${message}">Something went wrong.</h4>
	<p class="nrml-txt" th:text="${trace}">Why not try refreshing you page? Or you can <a href="https://www.mdaima.com">

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 注释的异常

SpringBoot如何自定义错误处理逻辑

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 &mdash;&mdash; 表示此次请求立刻结束,发出 /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();
    }
}

自定义异常处理器被加入(未调整优先级时,默认加到最后):

SpringBoot如何自定义错误处理逻辑

6. ErrorViewResolver实现自定义处理异常

交由ErrorViewResolver的情况 :

情况一: response.sendError ,error 请求转给 Controller

情况二: 判断异常请求无人处理,tomcat 底层 response.sendError

ErrorViewResolver处理方法:

BasicErrorController 得到要去的地址后由 ErrorViewResolver 解析,解析规则为拼接状态码等信息

&mdash;&mdash; 即 ErrorViewResolver 是最后的异常处理, 没人处理的异常,被它处理

相关文章