2016-09-25 84 views
2

我有一个JSON文件(即我无法控制的),看起来像这样:杰克逊JsonTypeInfo为HashMap中<字符串,对象>的内容

{ 
    "some-identifier": { 
    "@class": "some-prefix.ClassA", 
    "<classA-property1>": "value1", 
    "<classA-property2>": "value2", 
    }, 
    "some-other-identifier": { 
    "@class": "some-other-prefix.ClassB", 
    "<classB-property1>": <... possibly nested objects ...> 
    }, 
    <...> 
} 

(该CLASSA的属性和ClassB-属性是分别是ClassA和ClassB成员的实际名称。)

我想将其反序列化为HashMap(将每个标识符映射到实际对象),并且我想使用自定义的TypeIdResolver来确定要实例化的实际类(我可以从前缀和类名中确定)。这些对象本身应该使用默认的解串器进行反序列化。

经过大量的搜索,我无法完成这项工作。我需要一些方法来注释HashMap,以便为其内容设置JsonTypeInfo和JsonTypeIdResolver。到目前为止,我所见过的所有示例都有基于所有子类扩展的基本类型的注释。然而,就我而言,JSON中包含的类没有共同的父类(当然除了Object)。我想过使用mixin注释Object本身,但即使如此,它也会为包含的对象打破默认的反序列化,因为它会在所有子对象上出现@class属性。

这种情况下有解决方案吗?

回答

0

我想你可以通过启用默认类型信息的对象映射器这样实现:

new ObjectMapper().enableDefaultTyping(
     ObjectMapper.DefaultTyping.JAVA_LANG_OBJECT, 
     JsonTypeInfo.As.PROPERTY); 

下面是一个完整的例子:

public class JacksonDefaultTypeInfo { 
    static class Bean1 { 
     public String value; 

     Bean1() {} 

     public Bean1(final String value) { 
      this.value = value; 
     } 

     @Override 
     public String toString() { 
      return "Bean1{" + 
        "value='" + value + '\'' + 
        '}'; 
     } 
    } 

    static class Bean2 { 
     public int number; 

     Bean2() {} 

     Bean2(final int number) { 
      this.number = number; 
     } 


    } 

    public static void main(String[] args) throws IOException { 
     final Map<String, Object> map = new HashMap<>(); 
     map.put("bean1", new Bean1("string")); 
     map.put("bean2", new Bean2(123)); 

     final ObjectMapper objectMapper = new ObjectMapper() 
       .enableDefaultTyping(
         ObjectMapper.DefaultTyping.JAVA_LANG_OBJECT, 
         JsonTypeInfo.As.PROPERTY); 
     final String json = objectMapper 
       .writerWithDefaultPrettyPrinter() 
       .writeValueAsString(map); 

     System.out.println(json); 
     final Map<String, Object> result = objectMapper.readValue(
       json, 
       new TypeReference<Map<String, Object>>() {}); 
     System.out.println(result); 
    } 
} 

输出:

{ 
    "bean1" : { 
    "@class" : "stackoverflow.JacksonDefaultTypeInfo$Bean1", 
    "value" : "string" 
    }, 
    "bean2" : { 
    "@class" : "stackoverflow.JacksonDefaultTypeInfo$Bean2", 
    "number" : 123 
    } 
} 
{bean1=Bean1{value='string'}, bean2=Bean2{number=123}}