Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 300|回复: 0

只需一步,在Spring Boot中统一Restful API返回值格式与统一处理异常

[复制链接]
  • TA的每日心情
    奋斗
    2024-4-6 11:05
  • 签到天数: 748 天

    [LV.9]以坛为家II

    2034

    主题

    2092

    帖子

    70万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    705612
    发表于 2021-5-9 21:32:56 | 显示全部楼层 |阅读模式

    统一返回值

    在前后端分离大行其道的今天,有一个统一的返回值格式不仅能使我们的接口看起来更漂亮,而且还可以使前端可以统一处理很多东西,避免很多问题的产生。

    比较通用的返回值格式如下:

    public class Result<T> {
        // 接口调用成功或者失败
        private Integer code = 0;
        // 失败的具体code
        private String errorCode = "";
        // 需要传递的信息,例如错误信息
        private String msg;
        // 需要传递的数据
        private T data;
        ...
    }
    

    最原始的接口如下:

    	@GetMapping("/test")
        public User test() {
            return new User();
        }
    

    当我们需要统一返回值时,可能会使用这样一个办法:

    	@GetMapping("/test")
        public Result test() {
            return Result.success(new User());
        }
    

    这个方法确实达到了统一接口返回值的目的,但是却有几个新问题诞生了:

    • 接口返回值不明显,不能一眼看出来该接口的返回值。
    • 每一个接口都需要增加额外的代码量。

    所幸Spring Boot已经为我们提供了更好的解决办法,只需要在项目中加上以下代码,就可以无感知的为我们统一全局返回值。

    /**
     * 全局返回值统一封装
     */
    @EnableWebMvc
    @Configuration
    public class GlobalReturnConfig {
    
        @RestControllerAdvice
        static class ResultResponseAdvice implements ResponseBodyAdvice<Object> {
            @Override
            public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
                return true;
            }
    
            @Override
            public Object beforeBodyWrite(Object body, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
                if (body instanceof Result) {
                    return body;
                }
                return new Result(body);
            }
        }
    }
    


    而我们的接口只需要写成最原始的样子就行了。

    	@GetMapping("/test")
        public User test() {
            return new User();
        }
    

    统一处理异常

    将返回值统一封装时我们没有考虑当接口抛出异常的情况。当接口抛出异常时让用户直接看到服务端的异常肯定是不够友好的,而我们也不可能每一个接口都去try/catch进行处理,此时只需要使用@ExceptionHandler注解即可无感知的全局统一处理异常。

    @RestControllerAdvice
    public class GlobalExceptionHandler {
    
        private static final Logger LOG = LoggerFactory.getLogger(GlobalExceptionHandler.class);
    
        /**
         * 全局异常处理
         */
        @ExceptionHandler
        public JsonData handleException(HttpServletRequest request, HttpServletResponse response, final Exception e) {
            LOG.error(e.getMessage(), e);
            if (e instanceof AlertException) {//可以在前端Alert的异常
                if (((AlertException) e).getRetCode() != null) {//预定义异常
                    return new Result(((AlertException) e).getRetCode());
                } else {
                    return new Result(1, e.getMessage() != null ? e.getMessage() : "");
                }
            } else {//其它异常
                if (Util.isProduct()) {//如果是正式环境,统一提示
                    return new Result(RetCode.ERROR);
                } else {//测试环境,alert异常信息
                    return new Result(1, StringUtils.isNotBlank(e.getMessage()) ? e.getMessage() : e.toString());
                }
            }
        }
    
    }
    

    其中的AlertException为我们自定义的异常,因此当业务中需要抛出错误时,可以手动抛出AlertException

    以上就是统一处理返回值和统一处理异常的两步。
    转评赞就是最大的鼓励

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-5-20 04:28 , Processed in 0.071028 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表