Spring Boot 统一功能处理(二)

        本篇主要介绍Spring Boot统一功能处理中的统一数据返回格式。

目录

一、定义统一的返回类

 二、配置统一数据格式

三、测试配置效果

四、统一格式返回的优点 

五、源码角度解析String问题


一、定义统一的返回类

在我们的接口在处理请求时,返回的结果可以说是参差不齐,既可以是一个String类型的数据,又可以是一个Integer类型的数据,这样未免会显得我们的代码很不规范,并且这种不规范的代码还会增加前后端之间的交流成本。因此,我们可以设计一个统一的类来作为所有接口的返回结果。这个统一的类通常需要包含以下三个部分:

  • code:由我们自己定义的一个状态码
  • msg:响应结果的描述
  • data:响应的数据

下面我们通过代码来具体来实现一下这个类 :

这里可以使用泛型来作为data的参数,毕竟返回的数据类型是多种多样的。

下面我们来改造一下上一篇中所定义的login方法:

这里的状态码200代表请求成功,-1代表失败。

仔细观察可以发现,这串代码其实是有点冗余的,每次返回数据时都得设置一次Result的三个属性。因此,我们还可以对Result进行一些优化,具体如下:

这里我们将设置Result属性的代码进行了封装,因此接口在返回数据时,直接调用这里的方法即可(由于fail表示请求失败,请求失败了通常不会有返回的数据,因此fail没有设置data参数)。接下来我们再优化一下前面的login方法:

 

这样代码就看着简洁多了。

 二、配置统一数据格式

虽然我们这里已经定义了统一的数据返回类Result,但我们并不能保证所有接口都会以这个类作为返回结果,因此我们还需要借助Spring 来配置一下统一的数据返回格式。

首先我们需要创建一个类来实现ResposeBodyAdvice:

然后重写两个方法:

 其中supports是用来选择要进行统一返回处理的方法,如果直接返回true那就是所有接口都会进行这个统一处理,如果不想全部都统一处理,也可以通过下面的方式获取相关的类或者方法,并以此来判定是否要统一处理:

         //获取执行返回操作的类Class<?> declaringClass = returnType.getMethod().getDeclaringClass();//获取执行返回操作的方法Method method = returnType.getMethod();

 另一个方法beforeBodyWrite是对返回的数据具体进行统一处理的逻辑,其中body就是我们在接口里返回的数据。这里已经写好了一个大体逻辑,如果是我们前面定义的返回类直接返回,如果不是就把body封装到返回类里再返回。

最后我们还需要再类上加上一个@ControllerAdvice注解:

通过这个注解的实现可以发现,他是@Component的衍生注解,因此加了这个注解的类也会被加到Spring容器中。并且可以发现@ControllerAdvice还多了很多别的功能,这里就不过多介绍了。 

三、测试配置效果

接下来我们通过postMan来测试一下配置的效果。

我们先测试一下返回类型为Result的接口:

postMan请求后的结果:

通过结果可以发现响应符合我们的统一要求。

然后我们创建一个返回值不为Result的接口:

通过PostMan来访问可以发现结果也是符合预期的:

 

然后我们再试试返回集合:

通过postMan来访问一下可以发现结果正常:

接下来我们再看看返回对象,

首先我们创建一个User类然后对这个User类进行返回:

通过PostMan可以发现结果正确。

最后我们再来看一下返回类型是String类型:

 通过PostMan来访问可以发现报错了:

 我们再去后端看一下:

可以发现后端报了一个类型不匹配异常,至于为什么会报这个异常,后面会解释,我们先来看看解决方法。

首先我们得在项目中引入Jackson依赖,由于Spring web中内置了这个依赖,因此不需要自己特意去引入Jackson直接引入Spring web依赖即可。

然后使用Jackson提供的ObjectMapper类,由于Spring中已经自动添加了这个类的Bean,我们可以直接注入使用,然后利用Object将返回的Result转为字符串,由于使用objectmapper转换成字符串需要处理一个异常,因此在这里可以加上@SneakyThrows(使用这个注解可以将受检查异常转为非受检查异常(运行时异常),从而使我们不再需要去处理异常信息)具体代码如下:

我们再通过postMan来测试一下,可以发现,这次没有报错了,但返回类型不在是Json,而是text了:

四、统一格式返回的优点 

为什么要这么大费周章设置一个统一返回的格式呢。因为它具有如下一些优点:

  • 方便前端接受和处理后端响应的数据
  • 降低了前后端的沟通成本
  • 有利于项目统一数据的维护和修改
  • 有利于后端进行统一的规范和标准制定,以免后端返回一些乱七八糟的数据
五、源码角度解析String问题

接下来我们从源码角度来解释一下为什么当返回的数据类型是String时在进行统一返回处理时会出现类型不匹配异常。

Spring mvc在使用时,会自动注册一些自带的HttpMessageConverter来处理返回的数据,注册的先后顺序为:

ByteArrayHttpMessageConverter、StringHttpMessageConverter、SourceHttpMessageCoverter、AllencompassingFormHttpMessageConverter

在注册AllencompassingFormHttpMessageConverter时会去引入一些根据项目依赖去引入其它的拓展HttpMessageConvert,例如,项目中如果添加了Jackson则会引入MappingJackson2HttpMessageConverter。

这些HttpMessageConverter会按照注册的顺序形成一条调用链,如果有数据返回时,spring会对这条链按照先后顺序进行遍历 ,如果找到能处理当前返回数据的数据类型的HttpMessageConverter则会直接使用这个HttpMessageConverter来处理,当我们返回的数据类型是非String时,例如集合,对象等,由于前面的几个HttpMessageConverter都不能使用,所以会使用前面最后引入的MappingJackson2HttpMessageConverter来处理数据,而当返回的是String时,则能够使用StringHttpMessageConverter来进行处理,而StringHttpMessageConverter又在链中靠前的位置,因此在返回String时会使用StringHttpMessageConverter来进行处理。

接下来我们进到HttpMessageConverter处理数据的具体方法writeWithHttpMessageConvert来看一下:

打开方法实现往下翻可以看到这样一行代码

仔细观察可以发现这行代码就是在调用我们前面所设置beforeBodyWrite方法,由此可以推断出在执行完这行代码后body由String转换为我们前面所设置的Result类型了。

继续往下翻可以发现这样一行代码

 由于我们返回的是String,所以这里的converter应该是StringHttpMessageConverter类型的,我们进到write(这里使用的是StringHttpMessageConverter父类AbstarctHttpMessageConverter实现的write方法)方法来看一下:

可以发现在这里使用的是泛型参数t来接收body,因此没有出现类型匹配问题,在这个方法里调用了一个addDefaultHeaders方法,由于StringHttpMessageConverter重写了这个方法,因此在这里使用的是StringHttpMessageConverter实现的这个方法,我们进到里面去看一下:

进到里面可以发现这个方法采用的是String类型的s来接收我们前面的t,也就是body,但body已经在前面通过我们定义的beforeBdoyWriter方法由String转换为了Result类型,因此这里在接收参数时,发生了类型不匹配异常,从而也就产生了前面类型不匹配的报错。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/311671.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux-进程控制

目录 1.进程创建 2. 进程终止 2.1 进程退出的场景 2.2 进程常见退出方法 2.3 return返回终止 2.4 exit()和_exit() 3. 进程等待 3.1 进程等待的原因 3.2 wait​编辑 3.3 waitpid 3.4 status 4. 进程替换 4.1 替换原理 4.2 exec函数系列 1.进程创建 在linux中for…

【学习笔记】rt-thread

任务 创建好任务&#xff0c;不管是动态还是静态创建&#xff0c;任务的状态是init &#xff0c;通过start方法来启动任务&#xff1b;线程大小 设置小了&#xff0c;无法正常工作&#xff1f;显示占空间100% 启动过程 TODO 这是编译器特性&#xff1f; 因为RT-Thread使用编…

代码随想录-算法训练营day14【二叉树01:理论基础、递归遍历、迭代遍历、统一迭代】

代码随想录-035期-算法训练营【博客笔记汇总表】-CSDN博客 第六章 二叉树part01今日内容&#xff1a; ● 理论基础 ● 递归遍历 ● 迭代遍历 ● 统一迭代详细布置 理论基础 需要了解 二叉树的种类&#xff0c;存储方式&#xff0c;遍历方式 以及二叉树的定义 文章讲解&#x…

ARM_day7:实现三个按键中断

程序代码&#xff1a; mykey.h: #ifndef __MYKEY_H__ #define __MYKEY_H__ #include "stm32mp1xx_rcc.h" #include "stm32mp1xx_gpio.h" #include "stm32mp1xx_exti.h" #include "stm32mp1xx_gic.h" extern void printf(const char …

03_信号和槽

信号和槽 系统的信号和槽自定义信号和槽Lambda表达式 系统的信号和槽 下面我们完成一个小功能&#xff0c;上面我们已经学习了按钮的创建&#xff0c;但是还没有体现出按钮的功能&#xff0c;按钮最大的功能也就是点击后触发一些事情&#xff0c;比如我们点击按钮&#xff0c;…

FAGLL03H 新增自定义字段

1、SGLPOS_N_GL_CT、SGLPOS_N_CT两个结构新增自定义字段 2、执行t-code:HDBVIEWS 3、实施增强 FAGL_LIB 4、使用select data方法 5、代码示例: method IF_FAGL_LIB~SELECT_DATA.FIELD-SYMBOLS: <fs> TYPE any.FIELD-SYMBOLS <ls_data> TYPE any.F…

ctf.show_web13

上传一句话木马 1.php文件&#xff0c;显示 再改后缀为.jpg&#xff0c;显示错误文件大小 用dirsearch扫一下 备份文件.bak 下载文件源码 <?php header("content-type:text/html;charsetutf-8");$filename $_FILES[file][name];$temp_name $_FILES[file][tm…

腾讯清华联合提出图像到视频生成方法-Follow-Your-Click:点击图像并加上简单提示词就可让图像动起来!

Follow-Your-Click只需单击一次和简短的提示就可以让图像的某一部分动起来&#xff0c;还支持不同的动作表达&#xff0c;比如微笑&#xff0c;悲伤&#xff0c;跳舞…… 相关链接 论文链接&#xff1a;https://arxiv.org/abs/2403.08268 项目链接&#xff1a;https://github…

html 引入vue Element ui 的方式

第一种&#xff1a;使用CDN的方式引入 <!--引入 element-ui 的样式&#xff0c;--> <link rel"stylesheet" href"https://unpkg.com/element-ui/lib/theme-chalk/index.css"> <!-- 必须先引入vue&#xff0c; 后使用element-ui --> <…

基于Docker构建CI/CD工具链(六)使用Apifox进行自动化测试

添加测试接口 在Spring Boot Demo项目里实现一个简单的用户管理系统的后端功能。具体需求如下&#xff1a; 实现了一个RESTful API&#xff0c;提供了以下两个接口 &#xff1a; POST请求 /users&#xff1a;用于创建新的用户。GET请求 /users&#xff1a;用于获取所有用户的列…

凡泰极客亮相2024 亚马逊云科技出海全球化论坛,为企业数字化出海赋能

随着「不出海&#xff0c;即出局」登上热搜榜单&#xff0c;企业出海已成燎原之势&#xff0c;3月29日&#xff0c;2024 亚马逊云科技出海全球化论坛在深圳成功举办&#xff0c;凡泰极客创始人梁启鸿受邀出席&#xff0c;并以 「App 2.0&#xff1a;以SuperApp构建智能数字生态…

HADOOP大数据处理技术8-JavaSe

投入地跳舞 就像没有人在一旁看着你一样 2024/4/8 5&#xff09;方法重载&#xff1a;在方法中 方法名称相同 但参数列表不同 方法名称相同 但是参数类型或个数不一样 好处&#xff1a;好记 6&#xff09;super &#xff08;只在具有继承关系的子类中使用&#xff09; 作用&a…

JAVA基础07-封装,类加载原理以及分析(内有练习代码)

目录 封装的理解 概念 目的 权限修饰符 访问权限从大到小 如何快速定义一个标准的Java类 1.普通方法 2.快捷键 练习 static 定义 静态与非静态区分 使用静态与非静态的场合 成员变量和局部变量 成员变量 局部变量 例子讲解&#xff1a;两数交换 解决方法 变…

研发岗-统信UOS系统配置npm git等前端常用配置

第一步 获取root权限 配置环境等都需要用到root权限&#xff0c;所以我们先获取到root权限&#xff0c;方便下面的操作 下载软件 在UOS应用商店下载的所需应用 版本都比较低 安装node 官网下载了【arm64】的包&#xff0c;解压到指定文件夹&#xff0c;设置链接&#xff0…

如何降低漏测, 避免上线后出bug,6年测试心得分享

一、漏测原因总结 &#xff08;1&#xff09;需求评审质量低&#xff0c;需求设计简单、只是简单描述功能&#xff0c;功能逻辑较少   &#xff08;2&#xff09;需求变更频繁   &#xff08;3&#xff09;缺少需求分解&#xff08;sql 文档、用例设计&#xff09;   &…

Unity 左右折叠显示与隐藏UI的简单实现

要实现一个简单的UI左右折叠显示与隐藏&#xff0c;可以结合遮罩&#xff0c;通过代码控制UI区块的宽度和位移来实现。 具体可以按以下步骤实现&#xff1a; 1、新建一个Image组件&#xff0c;并添加精灵&#xff0c;调整大小后&#xff0c;复制一份作为该UI的父物体&#xf…

element UI 设置type=“textarea“ 禁止输入框缩放

背景 在 Element UI 中&#xff0c;当您使用 el-input 组件并设置 type"textarea" 时&#xff0c;默认情况下&#xff0c;用户可以通过拖动输入框的右下角来调整其大小。如果您想禁止这种缩放行为&#xff0c;需要使用 CSS 来覆盖默认的浏览器行为。 注意上图&#x…

WPS的JS宏如何实现全文件路径字符串中截取文件名(excel)

从全文件路径的字符串中&#xff0c;截取文件名称&#xff0c;例如&#xff1a; 全文件路径字符串为&#xff1a;C:\Windows\System32\drivers\acpi1.sys 需要截取文件名&#xff1a;acpi1.sys 方法如下&#xff1a; 1、简单的方式&#xff1a;把全文件路径字符串拷贝&…

[Linux - C] 自主Shell

[Linux - C] 自主Shell [Linux - C语言] 自主Shell逻辑策划 main()打印命令行 void MakeCommandLineAndPrint()用户名 USER主机名 HOSTNAME当前目录 PWDSkipPath 切割目录打印命令行 获取用户字符串 int GetUserCommand()检查重定向 void CheckRedir()切割字符 void SplitComma…

基于Springboot的餐厅点餐系统

基于SpringbootVue的餐厅点餐系统的设计与实现 开发语言&#xff1a;Java数据库&#xff1a;MySQL技术&#xff1a;SpringbootMybatis工具&#xff1a;IDEA、Maven、Navicat 系统展示 首页展示 菜品详情页 菜品信息 个人中心 后台管理 菜品信息管理 用户管理 菜…