飞不高 发表于 2025-1-2 21:59:46

springmvc-拦截器-异常处理

拦截器

   SpringMVC 内置拦截器机制 ,允许在请求被目标方法处理的前后进行拦截,执行一些额外操纵。
使用步调: ①实现 HandlerInterceptor 接口的组件即可成为拦截器
                    ②创建 WebMvcConfigurer 组件,并设置拦截器的拦截路径。
@Component
//拦截器
public class MyHandlerInterceptor implements HandlerInterceptor {

//    目标方法执行之前
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
      System.out.println("MyHandlerInterceptor.preHandle");
      return true; //true放行,false拦截
    }

//    目标方法执行完成之后
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, org.springframework.web.servlet.ModelAndView modelAndView) throws Exception {
      System.out.println("MyHandlerInterceptor.postHandle");
    }

//    页面渲染完后
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
      System.out.println("MyHandlerInterceptor.afterCompletion");
    }
}
/**
* 拦截器需要配置,告诉spring,这个拦截器主要拦截什么
*WebMvcConfigurer:专门对springmvc底层做一些配置
*容器需要有 WebMvcConfigurer组件
*   方式1:@Bean 放一个 WebMvcConfigurer
*   方式2:实现 WebMvcConfigurer接口
*/
@Configuration
public class MySpringMVCConfig implements WebMvcConfigurer {
    @Autowired
    private MyHandlerInterceptor myHandlerInterceptor;
//    添加拦截器
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
      registry.addInterceptor(myHandlerInterceptor).addPathPatterns("/**"); //拦截所有请求
    }
} 拦截器执行顺序
    ~当拦截器有多个时,执行顺序:
                   preHandle顺序执行——>postHandle逆序执行——>afterCompletion逆序执行。
 ~当postHandle、afterCompletion从哪炸炸了,倒序链路从哪结束。
 ~当某个preHandle返回false进行拦截时,
    postHandle是目标方法执行之后执行,preHandle返回true时,afterCompletion才会执行。
画图
拦截器or过滤器
https://i-blog.csdnimg.cn/direct/bbca43279fbe4eb3b08197c25a2c3862.png
异常处理

    ~异常处理:
           后端只编写正确的业务逻辑,如出现异常,通过抛异常方式提前停止业务逻辑,
            让前端感知异常。
~异常处理最终方式:
   1. 必须有业务异常类(区分异常),
   2. 必须有异常罗列类,(列举项目中每个模块出现的所有异常)
   3. 编写业务代码的时候,只需要编写正确业务逻辑,如果出现预期问题,
       使用抛异常方式(抛的时候把异常罗列一传),停止业务逻辑并关照上层controller
   4. GlobalExceptionHandler全局异常处理器捕获Controller层抛出的异常并处理,
       全局拿到业务异常code msg 最终给前端相应。
@Service
public class EmployeeServiceImpl implements EmployeeService {
    @Autowired
    private EmployeeDao employeeDao;

    @Override
    public void updateEmployee(Employee employee) {
      Long id=employee.getId();
      if (id==null){
            throw new BuzException(BizExceptionEnume.ORDER_NOT_EXIST); //抛到业务异常类
      }
      employeeDao.updateEmp(employee);
}
/**
* 业务异常类
*/
@Data
public class BuzException extendsRuntimeException {
    private Integer code; //异常码
    private String msg; //异常信息

//    只传异常枚举对象
    public BuzException(BizExceptionEnume exceptionEnume){
      super(exceptionEnume.getMsg());
      System.out.println("业务异常类");
      this.code = exceptionEnume.getCode();
      this.msg = exceptionEnume.getMsg();
    }
} /**
* 异常处理文档 枚举类
*将来修改的时候,很麻烦,得改很多地方,使用枚举类进行固化
*/
public enum BizExceptionEnume {
// 根据业务动态扩充
//    ORDER_xx订单模块相关异常
    ORDER_CLOSED(10001,"订单已关闭"),
    ORDER_NOT_EXIST(10002,"订单不存在"),
//    PRODUCT_xx商品模块相关异常
    PRODUCT_NOT_EXIST(10003,"库存不足"),
    PRODUCT_STOCK_ERROR(10004,"库存错误");
//    ...........

    @Getter
    private Integer code;
    @Getter
    private String msg;

    private BizExceptionEnume(Integer code, String msg) {
      this.code = code;
      this.msg = msg;
    }
} /**
* 全局异常处理- 只对controller层进行处理
* @RestControllerAdvice合成注解<——@ResponseBody(R以json方式返回)、@ControllerAdvice(告诉springmvc这是一个全局异常处理类)
*/

@RestControllerAdvice
public class GlobalExceptionHandler {

//    业务异常
    @ExceptionHandler(BuzException.class)
    public R handleBizException(BuzException e){
      return R.error(e.getCode(), e.getMessage());
    }

    //    最终的兜底异常处理方法
    @ExceptionHandler(Throwable.class) //处理异常错误
    public R error(Exception e){
      return R.error(500, e.getMessage());
    }
}


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: springmvc-拦截器-异常处理