如何使用Jackson将对象转换为使用默认键入的地图?

时间:2011-10-31 17:00:21

标签: java json jackson

我正在使用Jackson将JAXB带注释的对象序列化为地图对象。这里有一些代码来说明我的问题:

    public class Test {

    @XmlAccessorType(XmlAccessType.NONE)
    public static class Inner {
        @XmlAttribute
        public int foo;
    }

    @XmlAccessorType(XmlAccessType.NONE)
    public static class Outer {
        @XmlAttribute
        public String bar;

        @XmlElement
        public Inner in;
    }

    public static void main(String[] args) throws Exception {
        ObjectMapper mapper = getMapper();

        mapper.enableDefaultTyping();

        Inner in = new Inner();
        in.foo = 42;
        Outer out = new Outer();
        out.in = in;
        out.bar = "thecakeisalie";

        Object o = mapper.convertValue(out, TreeMap.class);
        System.out.println(o);
    }

    public static ObjectMapper getMapper() {
        ObjectMapper mapper = new ObjectMapper();
        AnnotationIntrospector introspector = new JaxbAnnotationIntrospector();
        mapper.setAnnotationIntrospector(introspector);
        return mapper;
    }
}

这会导致以下错误:

Exception in thread "main" java.lang.IllegalArgumentException: Unexpected token (START_OBJECT), expected START_ARRAY: need JSON Array to contain As.WRAPPER_ARRAY type information for class java.lang.Object
 at [Source: N/A; line: -1, column: -1]
    at org.codehaus.jackson.map.ObjectMapper._convert(ObjectMapper.java:2493)
    at org.codehaus.jackson.map.ObjectMapper.convertValue(ObjectMapper.java:2459)
    at com.example.test.Test.main(Test.java:49)
Caused by: org.codehaus.jackson.map.JsonMappingException: Unexpected token (START_OBJECT), expected START_ARRAY: need JSON Array to contain As.WRAPPER_ARRAY type information for class java.lang.Object
 at [Source: N/A; line: -1, column: -1]
    at org.codehaus.jackson.map.JsonMappingException.from(JsonMappingException.java:163)
    at org.codehaus.jackson.map.deser.StdDeserializationContext.wrongTokenException(StdDeserializationContext.java:261)
    at org.codehaus.jackson.map.jsontype.impl.AsArrayTypeDeserializer._locateTypeId(AsArrayTypeDeserializer.java:100)
    at org.codehaus.jackson.map.jsontype.impl.AsArrayTypeDeserializer._deserialize(AsArrayTypeDeserializer.java:86)
    at org.codehaus.jackson.map.jsontype.impl.AsArrayTypeDeserializer.deserializeTypedFromAny(AsArrayTypeDeserializer.java:69)
    at org.codehaus.jackson.map.deser.std.UntypedObjectDeserializer.deserializeWithType(UntypedObjectDeserializer.java:106)
    at org.codehaus.jackson.map.deser.std.MapDeserializer._readAndBind(MapDeserializer.java:321)
    at org.codehaus.jackson.map.deser.std.MapDeserializer.deserialize(MapDeserializer.java:249)
    at org.codehaus.jackson.map.deser.std.MapDeserializer.deserialize(MapDeserializer.java:33)
    at org.codehaus.jackson.map.ObjectMapper._readValue(ObjectMapper.java:2695)
    at org.codehaus.jackson.map.ObjectMapper.readValue(ObjectMapper.java:1308)
    at org.codehaus.jackson.map.ObjectMapper._convert(ObjectMapper.java:2489)
    ... 2 more

如果省略行mapper.enableDefaultTyping();,代码将工作并输出以下内容:

{bar=thecakeisalie, in={foo=42}}

如果我使用mapper序列化为json,它将使用默认输入。

默认输入是否与对象转换不兼容,还是我使用它错了?

1 个答案:

答案 0 :(得分:2)

确定。 Hmmh。问题源于一些不兼容的用法:当序列化时,默认类型只会根据其配置添加类型信息,并且在这种情况下不包括类型信息(类型不是抽象的,也不是声明为java.lang)。宾语)。 但是当反序列化(作为转换的一部分)时,结果类型需要类型信息,因为名义类型是TreeMap<Object,Object>;这就是异常的来源。

但是因为你正在转换为Map - 这实际上是一种“无类型”类型 - 你最好不要启用默认输入。无论如何,这种类型的信息都会被丢弃。

或者,如果你真的想看到包含的类型信息,你需要进行两阶段处理:启用默认类型的序列化(使用已启用它的mapper),并反序列化为TreeMap,使用具有默认类型的ObjectMapper禁用。