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

    1
    2
    3
    4
    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序列化工具

    1
    2
    3
    ObjectMapper om = new ObjectMapper();
    om.setVisibility(JsonMethod.ALL, JsonAutoDetect.Visibility.ANY);
    om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
  3. 序列化

    1
    2
    String json = om.writeValueAsString(map);
    System.out.println(json);
  4. 序列化输出

    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]]}]
  5. 反序列化

    1
    2
    Map<Integer,List<Integer>> map2 = om.readValue(json, Map.class);
    System.out.println(map2);
  6. 反序列化输出

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

    image

分析

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

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

解决

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

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

image

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

>