zoukankan      html  css  js  c++  java
  • Spring Boot中Restful Api的异常统一处理

    我们在用Spring Boot去向前端提供Restful Api接口时,经常会遇到接口处理异常的情况,产生异常的可能原因是参数错误,空指针异常,SQL执行错误等等。

    当发生这些异常时,Spring Boot会自动对异常进行一次统一的处理,返回一个异常信息:

    @RestController
    public class LiveRestController {
     
     
        // 用户查询某一种直播状态下的所有直播信息
        @GetMapping("/lives")
        public List<LiveResponse> getLiveList(@RequestParam Long status) {
            // 这里特意写上.toString()是为了触发NPE
            log.info("get status:{} lives", status.toString());
            return Lists.newArrayList();
        }
     
     
    }
    

    当不带参数向这个接口发起请求时就会得到下面的结果:
    MVyBOs.png

    这样的结果对于用户来说是很不友好的,前端也并不能够向用户展示这样的错误信息。

    我们可以看到这个异常的处理日志如下:

    2019-10-15 23:03:25.351  WARN 31571 --- [nio-8080-exec-1] .w.s.m.s.DefaultHandlerExceptionResolver : Resolved [org.springframework.web.bind.MissingServletRequestParameterException: Required Long parameter 'status' is not present]
    

    这个就是由Spring Boot内部异常处理类来处理的结果。对此,我们需要去实现定制自己的异常处理类,已实现更好的异常提示。

    首先要创建全局异常处理类,就需要用到两个重要的注解:

    @ControllerAdvice定义统一的异常处理类,这样就不必在每个Controller中逐个定义AOP去拦截处理异常。
    @ExceptionHandler用来定义函数针对的异常类型,最后将Exception对象处理成自己想要的结果。
    

    下面给出自定义异常处理的示例代码:

    // 注解定义异常统一处理类
    @ControllerAdvice
    public class RestfulApiExceptionHandler {
     
        // 这里处理MissingServletRequestParameterException的异常
        @ExceptionHandler(value = MissingServletRequestParameterException.class)
        // 返回JSON数据
        @ResponseBody
        // 特别注意第二个参数为要处理的异常
        public Map<String, Object> requestExceptionHandler(HttpServletRequest request, MissingServletRequestParameterException exception) {
            Map<String, Object> error = Maps.newHashMap();
            error.put("status", 500);
            // 在这里可以定义返回的异常提示信息
            error.put("message", "参数" + exception.getParameterName() + "错误");
            return error;
        }
     
        // 这里处理没有被上一个方法处理的异常,通常在最后的方法处理最大的Exception,保证兜底
        @ExceptionHandler(value = Exception.class)
        @ResponseBody
        public Map<String, Object> exceptionHandler(HttpServletRequest request, Exception exception) throws Exception {
            Map<String, Object> error = Maps.newHashMap();
            error.put("status", 500);
            error.put("message", "系统错误");
            return error;
        }
     
    }
    

    如此异常信息会被处理成:
    MVyImR.png

    可以看到异常信息被改成了我们自定义的形式,但是这个时候的HTTP状态码被处理成了200,我们可以通过定义的status来完成错误请求的识别与处理。

  • 相关阅读:
    MVC模式 与 Model2模型 介绍
    通过Servlet获取初始化参数
    《算法竞赛入门经典》学习笔记 2.3 文件操作
    《C++ Primer Plus》学习笔记 2.1.1 main()函数
    Codeforces Round #359 (Div. 2) C. Robbers' watch 搜索
    Github Pages建立个人博客
    《算法竞赛入门经典》第一章 程序设计入门 习题
    《C++ Primer Plus》学习笔记 第1章 预备知识
    在C#代码中应用Log4Net(一)简单使用Log4Net
    NHibernate实例
  • 原文地址:https://www.cnblogs.com/tingyugetc/p/11820259.html
Copyright © 2011-2022 走看看