Jackson序列化丢失泛型
经过
项目中遇到一个奇怪的bug,即一个Map<Integer,List<Integer>>
的泛型map,向map中get一个存在的key,事实上却返回null。
经过排查,发现是该map被Jackson序列化后,key的类型从Integer变成了String类型。再经过反序列化,即使已经声明key泛型的Integer,反序列化后内存数据中的key为String并不是Integer类型且并未抛出异常。
复现
声明一个key泛型为Integer的map
1
2
3
4Map<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));申明Jackson序列化工具
1
2
3ObjectMapper om = new ObjectMapper();
om.setVisibility(JsonMethod.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);序列化
1
2String json = om.writeValueAsString(map);
System.out.println(json);序列化输出
1
["java.util.HashMap",{"1":["java.util.ArrayList",[1,2,3]],"50001":["java.util.ArrayList",[7,8,9]],"1001":["java.util.ArrayList",[4,5,6]]}]
反序列化
1
2Map<Integer,List<Integer>> map2 = om.readValue(json, Map.class);
System.out.println(map2);反序列化输出
1
{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指定反序列化对象的泛型。
1 | Map<Integer, List<Integer>> map3 = om.readValue(json, new TypeReference<Map<Integer, List<Integer>>>(){}); |
对于通用型序列化反序列化的场景,例如RedisTemplate的序列化反序列化工具,无法指定特定的反序列化对象泛型,可以考虑使用其他序列化工具替代Jackson例如Fastjson。