我有一个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属性。
这种情况有解决方案吗?
答案 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}}