记录Jackson和Lombok的坑
记录Jackson和Lombok的坑
今天遇到Jackson反序列化json缺少了字段,后来研究下发现是Jackson的机制和Lombok生成的setter不一致,导致没有正确调用setter。
复现
Java实体类
@Data
public class DemoData{
private Double t;
private Double eDay;
}
Json字符串
{
"t":12.23,
"eDay":123.321
}
使用Jackson解析下来,发现只有t有值,而eDay没有解析到。
原因分析
首先第一反应是Lombok生成的getter和setter也许有问题,于是去掉@Data注解,用IDEA生成getter和setter,再进行反序列化,发现已经可以正常反序列化了。
于是看了下编译生成的代码:
public class DemoData{
private Double t;
private Double eDay;
public Double getT() {
return this.t;
}
public Double getEDay() {
return this.eDay;
}
public void setT(final Double t) {
this.t = t;
}
public void setEDay(final Double eDay) {
this.eDay = eDay;
}
}
去掉lombok的注解,直接用IDEA生成getter和setter,生成之后是这样的:
public class DemoData{
private Double t;
private Double eDay;
public Double getT() {
return t;
}
public void setT(Double t) {
this.t = t;
}
public Double geteDay() {
return eDay;
}
public void seteDay(Double eDay) {
this.eDay = eDay;
}
}
显然两边的Getter和Setter是不一样的,那么Jackson是怎么寻找属性和Setter的呢?
Jackson2在初始化序列器时,对pojo类型对象会收集其属性信息,属性包括成员变量及方法,然后属性名称和处理过后的方法名称做为key保存到一个LinkedHashMap中。
收集过程中会调用com.fasterxml.jackson.databind.util.BeanUtil中的legacyManglePropertyName方法用来处理方法名称,它会将get/set方法前缀,即get或set去掉,并将其后面的连续大写字符转换成小写字符返回。
例如: getNEWString会转变成newstring返回。你的属性名称如果有这样的"nSmallSellCount",lombok自动生成的get方法就会是这样的"getNSmallSellCount",处理过后就是这样的"nsmallSellCount",这与属性nSmallSellCount并不冲突,可以同时存在于HashMap中。
所以,当Jackson扫描由Lombok生成的POJO时,读取到setEDay,会把set去掉,拿到EDay,然后转成eday。由此导致json中的eDay属性在LinkedHashMap中没有找到setter方法,反序列化就丢失了字段。
所以原因已经确定了:当使用Lombok修饰的POJO中存在由aAxxx这样的(单个小写字母跟着大写字母)的属性时,反序列化会丢失这个字段。
如何解决
DeLombok
当代码中出现这样的字段时,由IDEA生成对应的getter和setter,会自动覆盖lombok生成的方法。
使用Builder来做Jackson的反序列化器
Lombok似乎意识到了这个问题(所以为啥不改下setter的生成呢???),编写了@Jacksonized这个注解来为Jackson反序列提供支持,但是这个注解必须配合@Builder或者@SuperBuilder一起使用才会生效。
我们看下@Jacksonized的官方说明:
/**
* The {@code @Jacksonized} annotation is an add-on annotation for
* {@code @}{@link Builder} and {@code @}{@link SuperBuilder}. It automatically
* configures the generated builder class to be used by Jackson's
* deserialization. It only has an effect if present at a context where there is
* also a {@code @Builder} or a {@code @SuperBuilder}; a warning is emitted
* otherwise.
* <p>
* In particular, the annotation does the following:
* <ul>
* <li>Configure Jackson to use the builder for deserialization using
* {@code @JsonDeserialize(builder=Foobar.FoobarBuilder[Impl].class)}
* on the class (where <em>Foobar</em> is the name of the annotated class).</li>
* <li>Copy Jackson-related configuration annotations (like
* {@code @JsonIgnoreProperties}) from the class to the builder class. This is
* necessary so that Jackson recognizes them when using the builder.</li>
* <li>Insert {@code @JsonPOJOBuilder(withPrefix="")} on the generated builder
* class to override Jackson's default prefix "with". If you configured a
* different prefix in lombok using {@code setterPrefix}, this value is used. If
* you changed the name of the {@code build()} method using using
* {@code buildMethodName}, this is also made known to Jackson.</li>
* <li>For {@code @SuperBuilder}, make the builder implementation class
* package-private.</li>
* </ul>
* This annotation does <em>not</em> change the behavior of the generated builder.
* A {@code @Jacksonized} {@code @SuperBuilder} remains fully compatible to
* regular {@code @SuperBuilder}s.
*/
简单来说,这个注解会做下面的事:
- 会通过
@JsonDeserialize注解让Jackson使用Builder来构建对象; - 拷贝Jackson相关的注解到Builder中(比如
@JsonIgnoreProperties); - 生成的Builder类会添加
@JsonPOJOBuilder注解并写入prefix;
因此,把上面的Pojo改写成这样:
@Data
@Builder
@Jacksonized
public class DemoData {
private Double t;
private Double eDay;
}
会生成下面的POJO:
@JsonDeserialize(
builder = DemoData.DemoDataBuilder.class
)
public class DemoData {
private Double t;
private Double eDay;
DemoData(final Double t, final Double eDay) {
this.t = t;
this.eDay = eDay;
}
public static DemoData.DemoDataBuilder builder() {
return new DemoData.DemoDataBuilder();
}
public Double getT() {
return this.t;
}
public Double getEDay() {
return this.eDay;
}
public void setT(final Double t) {
this.t = t;
}
public void setEDay(final Double eDay) {
this.eDay = eDay;
}
public boolean equals(final Object o) {
if (o == this) {
return true;
} else if (!(o instanceof DemoData)) {
return false;
} else {
DemoData other = (DemoData)o;
if (!other.canEqual(this)) {
return false;
} else {
Object this$t = this.getT();
Object other$t = other.getT();
if (this$t == null) {
if (other$t != null) {
return false;
}
} else if (!this$t.equals(other$t)) {
return false;
}
Object this$eDay = this.getEDay();
Object other$eDay = other.getEDay();
if (this$eDay == null) {
if (other$eDay != null) {
return false;
}
} else if (!this$eDay.equals(other$eDay)) {
return false;
}
return true;
}
}
}
protected boolean canEqual(final Object other) {
return other instanceof DemoData;
}
public int hashCode() {
int PRIME = true;
int result = 1;
Object $t = this.getT();
int result = result * 59 + ($t == null ? 43 : $t.hashCode());
Object $eDay = this.getEDay();
result = result * 59 + ($eDay == null ? 43 : $eDay.hashCode());
return result;
}
public String toString() {
Double var10000 = this.getT();
return "DemoData(t=" + var10000 + ", eDay=" + this.getEDay() + ")";
}
@JsonPOJOBuilder(
withPrefix = "",
buildMethodName = "build"
)
public static class DemoDataBuilder {
private Double t;
private Double eDay;
DemoDataBuilder() {
}
public DemoData.DemoDataBuilder t(final Double t) {
this.t = t;
return this;
}
public DemoData.DemoDataBuilder eDay(final Double eDay) {
this.eDay = eDay;
return this;
}
public DemoData build() {
return new DemoData(this.t, this.eDay);
}
public String toString() {
return "DemoData.DemoDataBuilder(t=" + this.t + ", eDay=" + this.eDay + ")";
}
}
}
此时,Jackson会使用建造者方法来构建对象,写入属性,Json也可以正常解析了。
记录Jackson和Lombok的坑的更多相关文章
- CozyRSS开发记录0-RSS阅读器开坑
CozyRSS开发记录0-RSS阅读器开坑 1.RSS RSS,全名是Really Simple Syndication,简易信息聚合. 关于RSS相关的介绍,网上可以很容易的找到.RSS阅读器是我几 ...
- lombok踩坑与思考
虽然接触到lombok已经有很长时间,但是大量使用lombok以减少代码编写还是在新团队编写新代码维护老代码中遇到的. 我个人并不主张使用lombok,其带来的代价足以抵消其便利,但是由于团队编码风格 ...
- Jackson中的那些坑
不符合驼峰规范的变量 “驼峰命名法”请自行百度.简单的来说就是变量的第一个单词以小写字母开始其他单词首字母大写,或者全部单词首字母都大写,分别称为“小驼峰”和“大驼峰” 比如一个符合驼峰规范命名的实体 ...
- 【bug记录】OS Lab4 踩坑记
OS Lab4 踩坑记 Lab4在之前Lab3的基础上,增加了系统调用,难度增加了很多.而且加上注释不详细,开玩笑的指导书,自己做起来困难较大.也遇到了大大小小的bug,调试了一整天. 本文记录笔者在 ...
- 【bug记录】OS Lab3 踩坑记
OS Lab3 踩坑记 Lab3在之前Lab2的基础上,增加了进程建立.调度和中断异常处理.其中测试包括进程建立以及进程调度部分. 由于是第一次做bug记录,而且是调试完bug后再做的记录,所以导致记 ...
- 记录使用Buildbot遇到的坑
Buildbot Tips Buildbot也是个大坑..我并不熟悉python,偏偏文档又少.这几天使用buildbot出了不少坑.有的解决了,有的绕过去,这里都把它们一一记下来. Force Bu ...
- 【Java】Jackson解析xml的坑
为了获取xml数据,在spring mvc中针对 @ResponseBody配置了jackson. 刚用的时候内心是狂喜的,终于不用自己解析了---- but----------还是有坑的-- 坑一 ...
- ios 记录支付宝集成遇到的坑及解决方法
今天项目中要开始动手集成支付宝支付,在此小结一下.(目前新版的支付宝SDK有较大改版,去集成还需要自己去开发平台详细的按照集成步骤来完成https://doc.open.alipay.com/docs ...
- 记录pageHelper分页orderby的坑
pageHelper的count查询会过滤查询sql中的order by条件! pageHelper分页功能很强大,如果开启count统计方法,在你执行查询条件时会再执行一条selet count(* ...
随机推荐
- Timer定时器开发
Timer定时器开发 定时器的作用是不占线程的等待一个确定时间,同样通过callback来通知定时器到期. 参考:https://github.com/sogou/workflow 定时器的创建 同样 ...
- Python分析离散心率信号(上)
Python分析离散心率信号(上) 一些理论和背景 心率包含许多有关信息.如果拥有心率传感器和一些数据,那么当然可以购买分析包或尝试一些可用的开源产品,但是并非所有产品都可以满足需求.也是这种情况.那 ...
- 实验4、Flask基于Blueprint & Bootstrap布局的应用服务
1. 实验内容 模块化工程内容能够更好的与项目组内成员合作,Flask Blueprint提供了重要的模块化功能,使得开发过程更加清晰便利.此外,Flask也支持Bootstrap的使用. 2. 实验 ...
- 【UG二次开发】 UF_OBJ_ask_name 获取对象名字
代码 char name[256]; UF_OBJ_ask_name(objTag, name);
- 身为一枚优秀的程序员必备的基于Redis的分布式锁和Redlock算法
1 前言 今天开始来和大家一起学习一下Redis实际应用篇,会写几个Redis的常见应用. 在我看来Redis最为典型的应用就是作为分布式缓存系统,其他的一些应用本质上并不是杀手锏功能,是基于Redi ...
- 安装Apache、Nginx和PHP-基于Centos7环境
使用的软件:putty或Xshell都可. 一.搭建Apache 1.编译安装 (1).安装编译器 yum install -y gcc (2)安装Opensll 查询官网得到OpenSSL下载网址h ...
- SCP,SSH应用
- 关于MySql数据库误操作数据找回的办法
先讲个事,前段时间,系统长时间不用的一个功能被开放出来了,想当然的我没有在测试平台上测试,直接操作了正式系统(的确是我不严谨),导致好多数据异常,页面展示错乱了.于是我想到的第一个就是进行备份还原.项 ...
- 复习Spring第二课--AOP原理及其实现方式
AOP原理: AOP,面向方面的编程,使用AOP,你可以将处理方面(Aspect)的代码注入主程序,通常主程序的主要目的并不在于处理这些aspect.AOP可以防止代码混乱.AOP的应用范围包括:持久 ...
- Unity使用Photon PUN2设置中国区服务器
原文地址:Unity使用Photon PUN2设置中国区服务器 入门系列 PUN2选择中国区服务器 先搜索中国区官网 选择试用购买 绑定你的Appid 注意: 当你的Appid申请了中国区后,海外的你 ...