当前位置: 首页 > news >正文

springmvc异常处理

springmvc异常处理

spring中有三种方式可以优雅的处理异常

  • 使用@ExceptionHandler
  • 使用HandlerExceptionResolver
  • 使用@ControllerAdvice+@ExceptionHandler

使用@ExceptionHandler

该方式只在指定的@Controller有效,不会对其他的@Controller产生影响

@Controller
@RequestMapping("/exception")
public class ExceptionController {

    // 使用@ExceptionHandler只对该@Controller有效,对其他Controller无效,如果想要对所有Controller生效,
    // 需要将该方法写到基类,让所有的Controller都继承该基类Controller
    @ExceptionHandler(BusinessException.class)
    @ResponseBody
    public String exception(Exception e)
{
        return "出现异常"+e.getMessage();
    }

    @RequestMapping("/testException")
    @ResponseBody
    public String testException(){
        User user = null;
        System.out.println(user.getId());
        return "success";
    }

    @RequestMapping("/testBusinessException")
    @ResponseBody
    public String testBusinessException(){
        throw new BusinessException();
    }
}

此时如果访问/exception/testBusinessException出现异常,就会跳转到exception方法中,将结果返回给浏览器

由于该方式只对@ExceptionHandler注解指定方法所在的Controller中生效,所以为了可以针对多个Controller生效,需要将@ExceptionHandler注解指定方法抽离到一个Controller基类中

@Controller
public class BaseController {
    @ExceptionHandler(BusinessException.class)
    @ResponseBody
    public String exception(Exception e)
{
        return "出现异常"+e.getMessage();
    }
}

然后需要该异常处理的Controller继承该基类

@Controller
@RequestMapping("/exception")
public class ExceptonController1 extends BaseController {

    @RequestMapping("/testBusinessException1")
    @ResponseBody
    public String testBusinessException(){
        throw new BusinessException();
    }
}

使用HandlerExceptionResolver

处理器异常解析器接口是负责处理各类控制器执行过程中出现的异常

public interface HandlerExceptionResolver {

   ModelAndView resolveException(
         HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
;

}

该方式是在DispatcherServlet中默认使用的,在processHandlerException()方法中,调用异常解析

protected ModelAndView processHandlerException(HttpServletRequest request, HttpServletResponse response,
      Object handler, Exception ex)
 throws Exception 
{

   // Check registered HandlerExceptionResolvers...
   ModelAndView exMv = null;
  // 异常处理器
   for (HandlerExceptionResolver handlerExceptionResolver : this.handlerExceptionResolvers) {
      exMv = handlerExceptionResolver.resolveException(request, response, handler, ex);
      if (exMv != null) {
         break;
      }
   }
   if (exMv != null) {
      if (exMv.isEmpty()) {
         request.setAttribute(EXCEPTION_ATTRIBUTE, ex);
         return null;
      }
      // We might still need view name translation for a plain error model...
      if (!exMv.hasView()) {
         exMv.setViewName(getDefaultViewName(request));
      }
      if (logger.isDebugEnabled()) {
         logger.debug("Handler execution resulted in exception - forwarding to resolved error view: " + exMv, ex);
      }
      WebUtils.exposeErrorRequestAttributes(request, ex, getServletName());
      return exMv;
   }

   throw ex;
}

其有四个实现类

DefaultHandlerExceptionResolver

DefaultHandlerExceptionResolver在DispatcherServlet中是默认使用的,用于将Spring中的标准异常解析为对应的HTTP状态码,但是响应体并不会改变

ResponseStatusExceptionResolver

ResponseStatusExceptionResolver在DispatcherServlet中是默认使用的,主要是和自定义异常上配置的@ResponseStatus注解进行搭配使用,将自定义异常映射到设定的HTTP状态码,与DefaultHandlerExceptionResolver一样,只是更改了状态码,并没有改变响应体

该异常处理机制是来解析@ResponseStatus来标注的异常

自定义异常

// code指定的是状态码,reason指定的是错误信息
@ResponseStatus(code = HttpStatus.BAD_REQUEST,reason = "出现业务异常")
public class BusinessException extends RuntimeException{

}
@RequestMapping("/testBusinessException")
@ResponseBody
public String testBusinessException(){
    throw new BusinessException();
}

调用该接口就会返回到状态码为400的错误页面

SimpleMappingExceptionResolver

SimpleMappingExceptionResolver用来映射异常类名到视图名

AnnotationMethodHandlerExceptionResolver

AnnotationMethodHandlerExceptionResolver通过注解@ExceptionHandler来处理异常,已经被废弃

ExceptionHandlerExceptionResolver
  • 如果出现异常,先是查找该Controller中用@ExceptionHandler注解定义的方法
  • 如果没有找到@ExceptionHandler注解的话,就会寻找标记了@ControllerAdvice注解的类中的@ExceptionHandler注解方法

除此之外,还可以自定义异常解析器,继承AbstractHandlerMethodExceptionResolver

自定义异常解析器
@Component
public class CustomExceptionResolver extends AbstractHandlerExceptionResolver {
    @Override
    protected ModelAndView doResolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
        if(ex instanceof BusinessException){
            System.out.println("出现业务异常");
            ModelAndView modelAndView = new ModelAndView();
            modelAndView.setStatus(HttpStatus.BAD_REQUEST);
            modelAndView.addObject("msg","出现业务异常");
            return modelAndView;
        }
        return null;
    }
}

但是由于该自定义解析依然返回的是ModelAndView,所以与目前前后端分离的项目不太搭

使用@ControllerAdvice+@ExceptionHandler

在前面的ExceptionHandlerExceptionResolver中已经用到了@ControllerAdvice,其实@ControllerAdvice是对于@ExceptionHandler的一个补充,使得可以进行全局的异常解析,可以将之前多个分散的@ExceptionHandler整合起来,合并成为一个单一的全局的异常处理中

@ControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler
    @ResponseBody
    public String exception(Exception e){
        return "全局捕获: 出现异常"+e.getMessage();
    }
}
  • 它允许对响应体和HTTP状态码进行完全控制
  • 它允许将几个异常映射到相同的方法,以便一起处理
  • 它充分利用了新的REST风格的 ResposeEntity响应
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface ControllerAdvice {

   
   @AliasFor("basePackages")
   String[] value() default {};

   // 指定生效的包
   @AliasFor("value")
   String[] basePackages() default {};

   // 指定生效的包
   Class<?>[] basePackageClasses() default {};

   // 指定生效的类
   Class<?>[] assignableTypes() default {};

   // 指定生效的注解,如RestController
   Class<? extends Annotation>[] annotations() default {};

}
局部异常处理

在当前Controller中处理异常(当前Controller中使用@ExceptionHandler标注的方法)

@Controller
@RequestMapping("/exception")
public class ExceptionController {

  /**
  * 在@Controller中所写的@ExceptionHandler方法只能处理该Controller类中出现的异常,不可以处理其他Controller中出现的异常,此为局部异常处理
  */

    @ExceptionHandler
    @ResponseBody
    public String exception(Exception e){
        return "出现异常"+e.getMessage();
    }

    @RequestMapping("/testException")
    @ResponseBody
    public String testException(){
        User user = null;
        System.out.println(user.getId());
        return "success";
    }
}
全局异常处理

如果当前Controller中没有异常处理,则会使用全局异常(使用@ControllerAdvice标注的类中的@ExceptionHandler方法)

@ControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler
    @ResponseBody
    public String exception(Exception e){
        return "全局捕获: 出现异常"+e.getMessage();
    }
}

也可以使用@RestControllerAdvice,相当于@ControllerAdvice+@ResponseBody

https://zhhll.icu/2021/框架/springmvc/基础/6.springmvc异常处理/

本文由 mdnice 多平台发布

http://www.lryc.cn/news/348672.html

相关文章:

  • 可拖动、连线的React画布组件有哪些? 官网分别是什么?
  • 专访 Staynex 创始人 Yuen Wong:酒店行业的变革者
  • 最新版Ceph( Reef版本)块存储简单对接k8s(上集)
  • 稳态大面积光伏组件IV测试太阳光模拟器
  • 编写HTTP协议代理的一些知识(源码)
  • LabVIEW天然气压缩因子软件设计
  • GCP谷歌云有什么数据库类型,该怎么选择
  • 项目经理之路:裁员与内卷下的生存策略
  • MWM触摸屏工控机维修TEM-EV0 EN00-Z312yy-xx
  • idm下载到99.99%不动了 idm突然不下载了 idm下载到最后没速度咋办 IDM下载后没网了是怎么回事
  • 设计模式-07 设计模式-观察者模式(Observer Pattern)
  • 戒烟网站|基于SSM+vue的戒烟网站系统的设计与实现(源码+数据库+文档)
  • 研发管理之认识DevOps
  • Spring MVC(五) 文件上传
  • Redis——Redis数据分片的三种算法
  • 【专利】一种日志快速分析方法、设备、存储介质
  • HFSS学习-day5-边界条件
  • spring Aop使用示例
  • MySQL-InnoDB数据存储结构
  • 【吊打面试官系列】Java高并发篇 - 什么是 Java Timer 类?如何创建一个有特定时间间隔的任务?
  • Spring生命周期深度解析
  • 基于 Windows 的记事本简单功能开发及部署发布--迭代2.0
  • Java lambda
  • 【智能算法】河马优化算法(HO)原理及实现
  • spring基础使用(案例)
  • 相同的树LeetCode
  • Vue中如何抽取部分代码到单独的ts文件
  • 山东齐鲁文化名人颜廷利:朱郭有文才,曲高‘菏’寡星光路
  • 嵌入式学习70-复习(wireshark使用和http协议)
  • idea配置MySQL提示