Jackson序列化丢失泛型

经过

项目中遇到一个奇怪的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。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值