Jackson JsonTypeInfo用于HashMap <String,Object>的内容 - java

我有一个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属性。

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

java参考方案

我认为您可以通过为对象映射器启用默认类型信息来实现此目的,如下所示:

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}}

合并List <T>和List <Optional <T >> - java

鉴于: List<Integer> integers = new ArrayList<>(Arrays.asList( 10, 12 )); List<Optional<Integer>> optionalIntegers = Arrays.asList( Optional.of(5), Optional.em…

实例化类型<?>的泛型类 - java

我正在为SCJP / OCPJP学习,并且遇到了一个对我来说很奇怪的示例问题。该示例代码实例化了两个通用集合:List<?> list = new ArrayList<?>(); List<? extends Object> list2 = new ArrayList<? extends Object>(); …

无法在Maven surefire中运行多个执行? - java

我想运行名称以ResourceTest.java结尾的测试类,因此我在执行后定义了它们。<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-surefire-plugin</artifactId> <co…

如何告诉Checker遗留方法将接受Nullable类型? - java

考虑一下:@Nullable Object obj = null; Optional<Object> optional = Optional.ofNullable(obj); 这会失败,因为检查器框架假定ofNullable无法接受null值(毕竟,其参数未标记为@Nullable)。有没有办法告诉Checker-framework这个方法(或我…

展平地图中的列表列表 - java

我有订单流(来源是订单列表)。每个订单都有一个客户和一个OrderLine列表。我要实现的目标是在一张简单的列表中以客户为关键的地图,并将属于该客户的所有订单行作为值。现在,通过执行以下操作,我现在管理的内容返回了Map<Customer>, List<Set<OrderLine>>>:orders .collect…