我的编程空间,编程开发者的网络收藏夹
学习永远不晚

SpringBoot怎么封装统一响应体

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

SpringBoot怎么封装统一响应体

这篇文章主要介绍了SpringBoot怎么封装统一响应体,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

一、前言

关于统一响应体的封装,没有一个标准答案,我在各种技术社区看了一遍,汇总了一个复用性比较好的方案。

二、添加结果类枚举

在项目目录下面建一个 responseEntity 的 package,然后在里面建一个 ResultEnum 枚举类,添加如下代码:

SpringBoot怎么封装统一响应体

这边介绍一下枚举类的用法。枚举类的作用实际上就是定义常量,如果不使用枚举类,通常采用静态常量来表示:

public static final Integer OK_CODE = 200;public static final String OK_MESSAGE = "成功";public static final Integer BAD_REQUEST_CODE = 400;public static final String BAD_REQUEST_MESSAGE = "参数错误";

这样的话存在一些问题,一是字段表意不明,特别是看别人的代码时,会很懵逼;第二当业务规模增大之后,可能要维护成百上千的静态常量,如果都写在一个文件里面,容易造成命名混淆,阅读也比较麻烦。

然后使用枚举类定义常量就比较方便,相当于一个接口,使用时只需要封装内部的数据类型,并且限定数据域。而且对于不同的枚举变量,可以调用不同的处理方法(实现枚举类的抽象方法可以做到这一点)。关于枚举类的一些知识点汇总如下:

  • 使用enum定义的枚举类默认继承了java.lang.Enum,实现了java.lang.Comparable接口,且不能继承其他类,也不可以被继承。但枚举类可以实现一个或多个接口;

  • 枚举类的所有实例必须放在第一行显示,不需使用new,不需显示调用构造方法,每个变量都是public static final修饰的,最终以分号结束。在之后的反编译中,我们就可以理解枚举类其实也是颗语法糖;

  • 枚举类的构造方法是私有的,默认的就是 private,所以不用再添加 private;

枚举类内部常用的方法:

  • valueOf() :返回当前枚举类的name属性,如果没有,则throw new java.lang.IllegalArgumentException();

  • values() :是编译器自动生成的方法,Enum中并没有该方法,返回包括所有枚举变量的数组;

  • toString()name() :两个方法一样,返回当前枚举类变量的name属性,如果觉得不够用,可以覆盖默认的 toString ,结合 SWITCH CASE 来灵活的实现 toString() 方法;

  • ordinal() :枚举类会给所有的枚举变量一个默认的次序,该次序从0开始,是根据我们定义的次序来排序的。而ordinal()方法就是获取这个次序(或者说下标);

  • compareTo() :比较的是两个枚举变量的次序,返回两个次序相减后的结果;

定义了枚举类之后,在类的上面添加 lombok 的 @Getter 注解,给对象的每个属性添加 getter 方法,方便后面获取常量。例如要获取 OK 的状态码,就可以这样写:

ResultEnum.OK.getCode()

这边再解释下 @Data@Getter@Setter 的区别:

  • @Data:注解在类上;提供类所有属性的 getter 和 setter 方法,此外还提供了equals、canEqual、hashCode、toString 方

  • @Getter:注解在属性上:为属性提供 getter 方法;注解再类上表示当前类中所有属性都生成getter方法

  • @Setter:注解在属性上:为属性提供 setter 方法;注解再类上表示当前类中所有属性都生成setter方法

三、添加统一结果类

还是在 responseEntity 目录下面,建一个 ServerResponse 类,添加如下代码:

@Datapublic class ServerResponse {private Boolean success;    private Integer code;    private String message;    private Object data;// 构造方法设为私有    private ServerResponse() {}public static ServerResponse ok(Object params) {        ServerResponse serverResponse = new ServerResponse();        serverResponse.setSuccess(ResultEnum.OK.getSuccess());        serverResponse.setCode(ResultEnum.OK.getCode());        serverResponse.setMessage(ResultEnum.OK.getMessage()); // 成功展示默认提示信息        serverResponse.setData(params); // 返回传入的参数        return serverResponse;    }public static ServerResponse badRequest(@Nullable String message) {        ServerResponse serverResponse = new ServerResponse();        serverResponse.setSuccess(ResultEnum.BAD_REQUEST.getSuccess());        serverResponse.setCode(ResultEnum.BAD_REQUEST.getCode());        serverResponse.setMessage(message != null ? message : ResultEnum.BAD_REQUEST.getMessage()); // 校验失败传入指定的提示信息        serverResponse.setData(null); // 校验失败不返回参数        return serverResponse;    }}

在上面的代码中,成员变量和构造方法都是私有的,只有静态方法向外暴露。然后处理成功的方法,message 展示默认提示信息,即定义在枚举类里面的常量,data 是需要传给前端的 JSON 参数;处理参数错误的方法,message 展示传进去的错误信息,如果传的是 null ,则展示默认提示信息,即定义在枚举类里面的常量,data 是传给前端的参数,但是在参数错误的情况下就不需要传了,因此是 null

这边有一个问题,暂时不清楚 Java 是否支持函数参数可选,本人测试发现定义函数的时候有参数,但是调用的时候不传,IDE 会给错误提示,因此这里通过传 null 来解决

四、控制层返回

在定义了统一结果类之后,就可以在接口中使用了。还是用之前那个方法,通过 POST 请求获取用户信息,再原封不动返回过去:

@PostMapping("validateUser")public ServerResponse userValidate(@RequestBody @Validated UserDTO userDTO) {    return ServerResponse.ok(null);}

这边先给参数传 null ,不给前端进行返回,看一下响应的结果:

SpringBoot怎么封装统一响应体

然后传递参数:

@PostMapping("validateUser")public ServerResponse userValidate(@RequestBody @Validated UserDTO userDTO) {    return ServerResponse.ok(userDTO);}

看一下响应的结果:

SpringBoot怎么封装统一响应体

五、异常处理类使用统一响应体

然后我们给异常处理的方法也添加统一响应体:

@RestControllerAdvicepublic class GlobalExceptionHandler {    @ExceptionHandler(value = MethodArgumentNotValidException.class)    public ServerResponse handleMethodArgumentNotValidException(MethodArgumentNotValidException e) {        // 注意传进去的表达式有可能是 null        // 因此在 ServerResponse 对 message 是否为 null 进行了判断        // 如果是 null 就展示默认的提示内容        return ServerResponse.badRequest(Objects.requireNonNull(e.getBindingResult().getFieldError()).getDefaultMessage());    }    // 其他异常处理方法}

然后我们模拟一下参数异常的情况:

SpringBoot怎么封装统一响应体

这样看起来是正常了,但是存在一个问题,后端判断参数异常的时候,因为我们捕获了异常,所以返回给前端的状态码还是 200 ,如何让状态码改为 400 呢?在 SpringBoot 中指定 HTTP 状态码主要有三种方式:

  • HttpServletResponse

  • @ResponseStatus

  • ResponseEntity

这边使用第三种方式,具体的用法看一下代码应该就明白了。我们把刚才统一结果类的方法修改下:

public static ResponseEntity<ServerResponse> badRequest(@Nullable String message) {ServerResponse serverResponse = new ServerResponse();    serverResponse.setSuccess(ResultEnum.BAD_REQUEST.getSuccess());    serverResponse.setCode(ResultEnum.BAD_REQUEST.getCode());    serverResponse.setMessage(message != null ? message : ResultEnum.BAD_REQUEST.getMessage()); // 校验失败传入指定的提示信息    serverResponse.setData(null); // 校验失败不返回参数    return new ResponseEntity<>(serverResponse, HttpStatus.BAD_REQUEST); // 使用 ResponseEntity 对象设置响应状态码}

可以看到我们用一个 ResponseEntity 对象包裹了我们封装的响应体,然后返回了这个对象。其中第二个参数就是状态码,HttpStatus.BAD_REQUEST 就代表 400 。然后我们还要修改下异常处理类的返回类型:

@RestControllerAdvicepublic class GlobalExceptionHandler {    @ExceptionHandler(value = MethodArgumentNotValidException.class)    public ResponseEntity<ServerResponse> handleMethodArgumentNotValidException(MethodArgumentNotValidException e) {        return ServerResponse.badRequest(Objects.requireNonNull(e.getBindingResult().getFieldError()).getDefaultMessage());    }    // 其他异常处理方法}

再来调试一下,这下状态码正常了:

SpringBoot怎么封装统一响应体

在统一结果类中所有的方法都根据上面的示例进行修改即可,我这边添加了几个方法,给各位参考下,具体可以根据业务场景进行添加:

@Datapublic class ServerResponse {    private Boolean success;    private Integer code;    private String message;    private Object data;    // 构造方法设为私有    private ServerResponse() {}        public static ResponseEntity<ServerResponse> ok(Object params) {        ServerResponse serverResponse = new ServerResponse();        serverResponse.setSuccess(ResultEnum.OK.getSuccess());        serverResponse.setCode(ResultEnum.OK.getCode());        serverResponse.setMessage(ResultEnum.OK.getMessage()); // 成功展示默认提示信息        serverResponse.setData(params); // 返回传入的参数        return new ResponseEntity<>(serverResponse, HttpStatus.OK);    }        public static ResponseEntity<ServerResponse> created() {        ServerResponse serverResponse = new ServerResponse();        serverResponse.setSuccess(ResultEnum.CREATED.getSuccess());        serverResponse.setCode(ResultEnum.CREATED.getCode());        serverResponse.setMessage(ResultEnum.CREATED.getMessage());//        serverResponse.setData(null);        return new ResponseEntity<>(serverResponse, HttpStatus.CREATED);    }        public static ResponseEntity<ServerResponse> noContent() {        return new ResponseEntity<>(null, HttpStatus.NO_CONTENT);    }        public static ResponseEntity<ServerResponse> badRequest(@Nullable String message) {        ServerResponse serverResponse = new ServerResponse();        serverResponse.setSuccess(ResultEnum.BAD_REQUEST.getSuccess());        serverResponse.setCode(ResultEnum.BAD_REQUEST.getCode());        serverResponse.setMessage(message != null ? message : ResultEnum.BAD_REQUEST.getMessage()); // 校验失败传入指定的提示信息//        serverResponse.setData(null); // 校验失败不返回参数        return new ResponseEntity<>(serverResponse, HttpStatus.BAD_REQUEST); // 使用 ResponseEntity 对象设置响应状态码    }}

此外,@ResponseStatus 也是一种设置状态码常用的方法,只需要在 Controller 方法中加一个注解就可以:

@PostMapping("validateUser")@ResponseStatus(code=HttpStatus.BAD_REQUEST, reason="参数异常")public ServerResponse userValidate(@RequestBody @Validated UserDTO userDTO) {    return ServerResponse.ok(userDTO);}

springboot是什么

springboot一种全新的编程规范,其设计目的是用来简化新Spring应用的初始搭建以及开发过程,SpringBoot也是一个服务于框架的框架,服务范围是简化配置文件。

感谢你能够认真阅读完这篇文章,希望小编分享的“SpringBoot怎么封装统一响应体”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网行业资讯频道,更多相关知识等着你来学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

SpringBoot怎么封装统一响应体

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

SpringBoot怎么封装统一响应体

这篇文章主要介绍了SpringBoot怎么封装统一响应体,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、前言关于统一响应体的封装,没有一个标准答案,我在各种技术社区看了一遍
2023-06-15

SpringBoot应用的启动入口怎么封装

这篇文章主要介绍了SpringBoot应用的启动入口怎么封装的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇SpringBoot应用的启动入口怎么封装文章都会有所收获,下面我们一起来看看吧。Springboot可
2023-06-30

vue3怎么封装input组件和统一表单数据

准备工作用vuecreateexample创建项目,参数大概如下:用原生input原生的input,主要是value和change,数据在change的时候需要同步。App.tsx如下:import{ref}from&#39;vue&#39;;exportdefault{setup(){//username就是数据constusername=ref(&#39;张三&#39;);//输入框变化的时候,同步数据constonInput=;return()=>({
2023-05-14

SpringBoot获取前台参数的方式及统一响应的方法是什么

本篇内容介绍了“SpringBoot获取前台参数的方式及统一响应的方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!请求SpringB
2023-07-05

win7系统程序未响应怎么办如何解决?具体步骤

很多win7用户碰到www.cppcns.com过这样的一个问题,就是提示说 “程序未响应&pythonrdquo;问题,而后程序就动不了啦!那么对于这个问题要怎么解决呢?现在一起去看看吧! 1.打开开始—所有程序
2023-06-02

使用CSS怎么实现一个响应式布局系统

这篇文章将为大家详细讲解有关使用CSS怎么实现一个响应式布局系统,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。什么是csscss是一种用来表现HTML或XML等文件样式的计算机语言,主要是用
2023-06-08

怎么使用Springboot封装一个自适配的数据单位转换工具类

今天小编给大家分享一下怎么使用Springboot封装一个自适配的数据单位转换工具类的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一
2023-07-05

怎么用代码实现一个迷你响应式系统vue

这篇文章主要讲解了“怎么用代码实现一个迷你响应式系统vue”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么用代码实现一个迷你响应式系统vue”吧!基本定义什么是响应式系统?学术上的定义,我
2023-07-05

Java中怎么实现实现一个简单得数据响应系统

这篇文章主要介绍“Java中怎么实现实现一个简单得数据响应系统”,在日常操作中,相信很多人在Java中怎么实现实现一个简单得数据响应系统问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java中怎么实现实现一个
2023-06-25

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录