Jackson JsonTypeInfo用于HashMap的内容<string,object =“”>

时间:2016-09-25 20:21:37

标签: java json jackson deserialization json-deserialization

我有一个JSON文件(我无法控制),如下所示:

{
  "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-properties和classB-properties分别是ClassA和ClassB成员的实际名称。)

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

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

这种情况有解决方案吗?

1 个答案:

答案 0 :(得分: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}}