spring中,为什么前端明明传了值,后端却接收不到

文章目录

  • 问题场景
  • 问题重现
  • 解决方式
  • 原因分析
  • 原理分析
  • 结论
  • 扩展

问题场景

在进行前后端的联调时,有时候会出现,前端明明传了值,后端接口却接收不到的情况,这种情况常常让人很苦恼,然后就会去仔细对比前后端的参数单词是不是对应上了,也会去检查是不是前端的请求参数格式有问题,又或者是后端接口接收的参数格式有问题,一通检查对比下来,发现都没问题。那究竟是为什么呢?那就继续往下看吧。


问题重现

控制层代码:

    @PostMapping(value = "/test")public void test(@RequestBody UserVO userVO) {System.out.println("用户代码:" + userVO.getUCode());System.out.println("用户名称:" + userVO.getUName());}

参数实体类:UserVO

@Data
public class UserVO {/*** 用户代码*/private Long uCode;/*** 用户名称*/private String uName;}

用postman模拟前端调用:

在这里插入图片描述
控制台预期打印结果:

用户代码:12345
用户名称:小明

控制台实际打印结果:
在这里插入图片描述


解决方式

在实体类的属性上方加@JsonProperty注解,如下图:

在这里插入图片描述

然后测试控制台打印结果:
在这里插入图片描述


原因分析

首先我们先把实体类复原,并且加上一个新的属性loginType

@Data
public class UserVO {/*** 用户代码*/private Long uCode;/*** 用户名称*/private String uName;/*** 登录类型*/private String loginType;}

眼尖的同学可能会发现了,我新加的属性loginType长得是不是跟原来两个属性uCode和uName不太一样,不一样的点在于uCode和uName都是首字母小写,第二个字母大写的单词,而loginType则不然。但是它们三都符合驼峰命名法的规范,对吧。这时候可以猜测,难道是这个原因导致的?

在这里我们先来简单验证下uCode、uName、loginType的情况

在这里插入图片描述

在这里插入图片描述
通过断点发现,uCode、uName是空的,loginType却不是空的
然后我们将uCode、uName分别改为userCode、userName后再进行测试

@Data
public class UserVO {/*** 用户代码*/private Long userCode;/*** 用户名称*/private String userName;/*** 登录类型*/private String loginType;}

在这里插入图片描述

在这里插入图片描述

这个时候我们就可以得出结论,原因就是首字母小写,第二个字母大写的单词的属性是有问题的。

但是我们不禁要问,为啥呢?它这也符合驼峰命名法的规范啊。为什么它就有问题呢?感兴趣的同学可以接着往下看。


原理分析

首先我们要知道,在Spring中,前后端之间数据传输会涉及到数据的序列化和反序列化的操作,并且SpringBoot默认是使用Jackson作为JSON数据格式处理的类库。

序列化:按照指定的格式、顺序等将实体类对象转换为JSON对象;
反序列化:将JSON对象中的字符串、数字等,将其转换为实体对象;

那么现在咱们就来断点调试Jackson的源码来看看原因。为方便展示,我将实体类留下uName、loginType两个属性

@Data
public class UserVO {/*** 用户名称*/private String uName;/*** 登录类型*/private String loginType;}

开始调试:
Jackon主要是通过抽象类AbstractJackson2HttpMessageConverterreadJavaType方法将 HTTP 请求中的消息体转换为对象,所以我们找到这部分代码,对他进行断点调试:

在这里插入图片描述
然后逐步断点,在上图的第192行和第195行,它会调用ObjectMapper.readValue,然后断点推进到调用方法的核心地方ObjectMapper_readMapAndClose方法

在这里插入图片描述
this._findRootDeserializer(ctxt, valueType);的大概意思就是根据类型找到反序列化器,注意在这边是先从缓存中取,取到了的话就直接返回了。如果没到下一步断点,在这边你可以清除一下缓存。
在这里插入图片描述
然后断点继续推进到创建反序列化器的地方DeserializerCache._createDeserializer

如果你清除缓存或者重启项目在调用时会直接进入到这个创建反序列化器的地方,你直接在这个方法上打断点就好了

在这里插入图片描述
找到上图中第164行的代码,BeanDescription是类的描述的意思,所有的属性都在这里被解析,然后我们断点进去看看。会进入到POJOPropertiesCollector.collectAll方法,就是字面意思,收集所有。方法逻辑详见下图:
在这里插入图片描述
执行完this._addFields(props);props加入了uNameloginType

在这里插入图片描述
执行完this._addMethods(props);后发现props竟然多了一个uname

在这里插入图片描述

在这里我们点开属性详细去看,会发现uName的get和set为空,但是loginType的是正常的,并且uname这个不知道哪里跑出来的属性的get和set也是不为空的。

在这里插入图片描述
在这里插入图片描述

再接着执行this._removeUnwantedProperties(props);移除不想要的属性之后,会发现就剩下loginTypeuname了,因为uName没有get和set。为什么
在这里插入图片描述
然后props中目前存储的就是loginTypeuname

现在我们就要弄明白为什么有get/set的是uname而不是uName

首先,在这个例子中我使用的是@Data注解,也就是使用的 Lombok,也就是说 getter 和 setter 是由 Lombok 生成的。使用注解的话会将get/set方法隐藏起来,然后我们可以通过IDEA的Structure来看,见下图:

在这里插入图片描述

那么Jackson 到底是如何解析的,使得解析出来的是uname,而不是uName。它解析的具体代码在com.fasterxml.jackson.databind.util.BeanUtil类中的legacyManglePropertyName方法中
在这里插入图片描述

从上图为我们可以很明显的看到,通过这个方法之后getLoginType被解析成loginType了。那我们再来看看uName,见下图:
在这里插入图片描述
从上图断点我们可以清晰的看见getUName被解析成uname了,按照我们正常的思维逻辑的话,loginType和uName都符合驼峰命名法的规范,那么uName对应的get方法解析出来应该是uName啊,为什么变成了uname呢?原因就在于这个legacyManglePropertyName方法的处理逻辑,它的逻辑大概是:

1.根据入参offset去除get或者get,然后就剩下UName或者LoginType了
2.然后从第一个字母开始解析,如果第一个字母是大写的,于是就将它转成小写,然后找下一个,如果还是大写,就继续转成小写,直到找到一个小写字母后,就把之后的字母(不管大小写)一起拼接进来。

这样就能解释了:

去除get之后的LoginType找到第一个字母是大写,转为小写的l,下一个字母是小写的了,就直接把后面的全拼接进来,最终形成了loginType

去除get之后的UName找到第一个字母是大写,转为小写的u,下一个字母又是大写,转为小写的n,在下一个字母是小写的了,就直接把后面的全拼接进来,最终形成了uname

如果说这边的getUName换成getuName,那么解析出来的就是正确的uName了。

结论

到这里,我们就可以得出结论了

因为 Lombok 生成 get、set 方法的语义规范与和Jackson 处理 get、set 方法之间的不一致,导致属性名无法匹配上,最终也就导致了前端明明传了参数,后端却接收不到的问题。

扩展

我后面去github的 lombok社区 了解了相关内容,lombook社区是这样描述的:

在这里插入图片描述
用网页翻译给他翻译成中文,翻译有些不对,但是能看明白大概意思就行

在这里插入图片描述

lombok的大概意思就是:我就是这样的规范,即使其他的工具框架都改了,我也不改,但是建议你们不要使用首字母小写第二个字母大写的属性名,避免出现问题,可能知名度比较高的框架都比较傲娇吧哈哈。

但是lombok还是给出了一个解决方案,加上这个配置项

lombok.accessors.capitalization = [basic | beanspec] (default: basic)

其中basic代表遵循lombok的规范(getUName);beanspec代表遵循Spring、Jackson 的规范(getuName)。默认是basic。

看到这里,我就来总结一下能解决这个问题的三种方案吧

1. 加@JsonProperty注解强行指定属性名

@Data
public class UserVO {/*** 用户名称*/@JsonProperty(value = "uName")private String uName;/*** 登录类型*/private String loginType;}

2.不使用lombok,使用IDEA默认生成get/set方法

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

3.加上lombok配置项

lombok.accessors.capitalization = [basic | beanspec] (default: basic)

最后,博主的建议是,尽量不要用这种命名方式,如果非要用,那就加上@JsonProperty注解强行指定属性名,这样比较方便。

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

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

相关文章

华为数通方向HCIP-DataCom H12-821题库(多选题:201-220)

第201题 以下关于BGP中Orginator ID属性的描述,正确的是哪些项? A、Originator ID属于公认任意属性 B、当其他BGP Speaker接收到这条路由的时候,将比较收到的0nginator ID和本地的Router ID,如果两个ID相同BGP Speaker会忽略掉这条路由,不做处理 C、当一条路由第一次被RR…

深度学习基础入门:从数学到实现

I. 引言 A. 深度学习的背景 深度学习是机器学习的一个重要分支,是一种基于神经网络的算法,被广泛应用于计算机视觉、自然语言处理、语音识别等领域。与传统机器学习算法相比,深度学习具有更高的容错性、复杂性和精度,需要庞大的…

短信系统后台搭建要注意什么|网页版短信平台开发

在搭建短信系统后台时,需要注意以下几个关键方面,以确保系统的稳定性、安全性和高效性: 选择合适的技术栈:根据项目需求和团队实际情况选择合适的后端开发语言和框架,如Java Spring、Node.js、Python Django等。 系统…

数字化转型核心:实现业务与技术深度融合的运维数字化管理之道

写在前面 数字化转型已经成为大势所趋,各行各业正朝着数字化方向转型,利用数字化转型方法论和前沿科学技术实现降本、提质、增效,从而提升竞争力。 数字化转型是一项长期工作,包含的要素非常丰富,如数字化转型顶层设…

输入与输出

输入(Scanner类) Scanner是java5的新特性,在java.util包里,可以完成用户输入。步骤: 导入java.util包;构造Scanner对象,参数为u标准输入流System.in;使用next()方法系列接收数据 nextBoolean()接收一个布…

AJAX-综合

文章目录 同步代码和异步代码回调函数地狱解决回调函数地狱Promise-链式调用async函数和awaitasync函数和await-捕获错误 事件循环宏任务与微任务Promise.all静态方法 同步代码和异步代码 同步代码:逐步执行,需原地等待结果后,才继续向下执行…

【Java核心能力】了解 Redis 的 IO 多路复用吗?

欢迎关注公众号(通过文章导读关注:【11来了】),及时收到 AI 前沿项目工具及新技术的推送! 在我后台回复 「资料」 可领取编程高频电子书! 在我后台回复「面试」可领取硬核面试笔记! 文章导读地址…

蓝桥杯小白月赛3.23

题目描述&#xff1a; AC代码&#xff1a; #include <iostream> #include<cstring> #include<algorithm>using namespace std;const int N 2e510; string str[N]; //写上&会速度更快一些 bool cmp(const string &s1,const string &s2) {//例…

VMware扩容硬盘

最近研究Oracle的备份导入导出功能&#xff0c;但是因为磁盘容量不够导致表空间的扩容没办法&#xff0c;从而没办法导入数据库的dmp文件。得想办法先扩容磁盘容量。话不多说上截图操作。 操作环境&#xff1a;VMware10 , Centos 6.9 VMware扩容硬盘步骤 一、关闭虚拟机&…

大学教材《C语言程序设计》(浙大版)课后习题解析 | 第三、四章

概述 本文主要提供《C语言程序设计》(浙大版) 第三、四章的课后习题解析&#xff0c;以方便同学们完成题目后作为参考对照。后续将更新第五、六章节课后习题解析&#xff0c;如想了解更多&#xff0c;请持续关注该专栏。 专栏直达链接&#xff1a;《C语言程序设计》(浙大版)_孟…

考研数学|《1800》《1000》《880》《660》最佳搭配使用方法

直接说结论&#xff1a;基础不好先做1800、强化之前660&#xff0c;强化可选880/1000题。 首先&#xff0c;传统习题册存在的一个问题是题量较大&#xff0c;但难度波动较大。《汤家凤1800》和《张宇1000》题量庞大&#xff0c;但有些题目难度不够平衡&#xff0c;有些过于简单…

带你学会深度学习之优化算法 - 1

前言 笔者写下此系列文章是希望在复习人工智能相关知识同时为想学此技术的人提供一定帮助。 图源网络&#xff0c;所有者可随时联系笔者删除。 在一个深度学习问题中&#xff0c;我们通常会预先定义一个损失函数。有了损失函数以后&#xff0c;我们就可以使用优化算法试图将其…