前端框架选型是企业提升开发效率与用户体验的关键因素
2577
2023-01-08
Jackson序列化丢失泛型的解决
Jackson序列化丢失泛型
经过
项目中遇到一个奇怪的bug,即一个Map
经过排查,发现是该map被Jackson序列化后,key的类型从Integer变成了String类型。再经过反序列化,即使已经声明key泛型的Integer,反序列化后内存数据中的key为String并不是Integer类型且并未抛出异常。
复现
1、声明一个key泛型为Integer的map
Map
map.put(1, Arrays.asList(1,2,3));
map.put(1001,Arrays.asList(4,5,6));
map.put(50001,Arrays.asList(7,8,9));
2、申明Jackson序列化工具
ObjectMapper om = new ObjectMapper();
om.setVisibility(jsonMethod.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
3、序列化
String json = om.writeValueAsString(map);
System.out.println(json);
4、序列化输出
["java.util.HashMap",{"1":["java.util.ArrayList",[1,2,3]],"50001":["java.util.ArrayList",[7,8,9]],"1001":["java.util.ArrayList",[4,5,6]]}]
5、反序列化
Map
System.out.println(map2);
6、反序列化输出
{1=[1, 2, 3], 50001=[7, 8, 9], 1001=[4, PowHCVvz5, 6]}
分析
由步骤4见得Map
且步骤6反序列化后,尽管map的key申明为Integer类型,但是Jackson反序列化后,依然将key反序列化为String类型,且未抛出任何异常。此时通过Integer的key获取map对应的值永远返回null。
解决
对于可以指定返回类型的反序列化,可以通过Jackson的API指定反序列化对象的泛型。
Map
System.out.println(map3);
对于通用型序列化反序列化的场景,例如RedisTemplate的序列化反序列化工具,无法指定特定的反序列化对象泛型,可以考虑使用其他序列化工具替代Jackson例如Fastjson。
序列化后反序列化丢失几大问题总结
序列化 (Serialization)将对象的状态信息转换为可以存储或传输的形式的过程。在序列化期间,对象将其当前状态写入到临时或持久性存储区。
以后,可以通过从存储区中读取或反序列化对象的状态,重新创建该对象。
反序列化失败原因:(目前只遇到过两种)
没有添加 serialVersionUID 可能会导致反序列化失败
生成默认的serialVersionUID --> [Add default serial version ID]
例如:
private static final long serialVersionUID = 1L;
生成串行serialVersionUID --> [Add generated serial version ID]
例如:
private static final long serialVersionUID =-5666638870709238304L;
注解生成serialVersionUID --> [Add @SuppressWarhttp://nings serial to serialVersionUID]
例如:
@SuppressWarnings("person")
public class Person implements Serializable {}
继承了一个已经实现序列化接口的父类
并且与父类有重复的属性,在反序列化的时候就会导致重复的属性数据丢失
然后还有第三种就是使用Spring框架的情况下
如果布尔类型的对象属性名以is开头,在序列化的时候会导致该属性值丢失
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~