Jackson序列化丢失泛型

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/sz85850597/article/details/82287461

经过

项目中遇到一个奇怪的bug,即一个Map<Integer,List<Integer>>的泛型map,向map中get一个存在的key,事实上却返回null。

经过排查,发现是该map被Jackson序列化后,key的类型从Integer变成了String类型。再经过反序列化,即使已经声明key泛型的Integer,反序列化后内存数据中的key为String并不是Integer类型且并未抛出异常。

复现

1 . 声明一个key泛型为Integer的map

Map<Integer, List<Integer>> map = new HashMap<>();
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<Integer,List<Integer>> map2 = om.readValue(json, Map.class);
System.out.println(map2);

6 . 反序列化输出

{1=[1, 2, 3], 50001=[7, 8, 9], 1001=[4, 5, 6]}

这里写图片描述

分析

由步骤4见得Map<Integer,List<Integer>>序列化后,key的Integer泛型已经丢失,类型由Integer变为了String。

且步骤6反序列化后,尽管map的key申明为Integer类型,但是Jackson反序列化后,依然将key反序列化为String类型,且未抛出任何异常。此时通过Integer的key获取map对应的值永远返回null。

解决

对于可以指定返回类型的反序列化,可以通过Jackson的API指定反序列化对象的泛型。

Map<Integer, List<Integer>> map3 = om.readValue(json, new TypeReference<Map<Integer, List<Integer>>>(){});
System.out.println(map3);

这里写图片描述

对于通用型序列化反序列化的场景,例如RedisTemplate的序列化反序列化工具,无法指定特定的反序列化对象泛型,可以考虑使用其他序列化工具替代Jackson例如Fastjson。

猜你喜欢

转载自blog.csdn.net/sz85850597/article/details/82287461