2016-09-25 2 views
2

У меня есть файл в формате JSON (что я не имею никакого контроля над), который выглядит следующим образом:Джексон JsonTypeInfo для содержания HashMap <String, Object>

{ 
    "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 ...> 
    }, 
    <...> 
} 

(The CLASSA-свойство и ClassB-свойства являются фактические имена членов ClassA и ClassB соответственно.)

Я хотел бы десериализовать это в HashMap (сопоставление каждого идентификатора с фактическим объектом), и я хочу использовать собственный TypeIdResolver для определения фактического класса, который будет создан (который я могу определить из префикса и имени класса). Затем сами объекты следует десериализовать, используя десериализатор по умолчанию.

После долгих поисков я не смог выполнить эту работу. Мне нужно каким-то образом аннотировать HashMap, чтобы установить JsonTypeInfo и JsonTypeIdResolver для его содержимого. Все примеры, которые я видел до сих пор, имеют те аннотации по базовому типу, из которых простираются все подклассы. Однако в моем случае нет общего родительского класса для классов, содержащихся в JSON (за исключением Object of course). Я подумал о том, что аннотировать сам объект с помощью mixin, но даже тогда это нарушит десериализацию по умолчанию для содержащихся объектов, так как тогда он ожидает свойство @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}}