杰克逊XML反序列化内联数组

13
如何反序列化这样奇怪的XML。在我看来,缺少props-entity(在props周围),但我无法更改此XML的源(一个Web服务)。
<parents>
  <parentname="first">
    <description><![CDATA[Description for the first-Entity]]></description>
    <prop name="level">
      <value><![CDATA[1]]></value>
    </prop>
    <prop name="enabled">
      <value><![CDATA[true]]></value>
    </prop>
    <prop name="version">
      <value><![CDATA[1.0-beta3]]></value>
    </prop>
  </parent>
  <parentname="second">...</parent>
  ...
</parents>

我的实体是:

public class Test {
    @Test
    public void deserializerTest() throws JsonParseException, JsonMappingException, IOException {
        ObjectMapper om = new XmlMapper();
        List<Parent> xml = om.readValue(new File("./test.xml"),
            new TypeReference<List<Parent>>() {});
    }
}

public class Prop {
    @JacksonXmlProperty(isAttribute = true)
    public String name;

    @JacksonXmlText
    public String value;
}

@JacksonXmlRootElement
public class Parent {
    @JacksonXmlProperty(isAttribute = true)
    public String name;

    public String description;

    // 1. alternative with List
    public List<Prop> prop;

    // 2. alternative with Map
    @JsonDeserialize(using = PropDeser.class)
    public Map<String, String> prop;
} 


public static class PropDeser extends JsonDeserializer<Map<String, String>> {

    @Override
    public Map<String, String> deserialize(JsonParser jp,
            DeserializationContext ctxt) throws IOException,
            JsonProcessingException {
        Map<String, String> ret = new HashMap<String, String>();
        boolean eof = false;
        while (jp.hasCurrentToken()) {
            JsonToken t = jp.getCurrentToken();
            switch (t) {
            case END_OBJECT:
                if (eof) {
                    return ret;
                }
                eof = true;
                break;
            case VALUE_STRING:
                ret.put(jp.getCurrentName(), jp.getText());
                break;
            default:
                eof = false;
                break;
            }
            jp.nextValue();
        }
        return null;
    }

}

1. 另一种选择

创建一个异常: “无法通过JSON字符串实例化类型为[简单类型,类my.test.Prop]的值;没有单个字符串构造函数/工厂方法(通过引用链:my.test.Parent [“prop”])”。

我不想要一个简单的字符串列表。我需要同时包含名称和值。所以我想到了使用 Map<String, String> 并创建自己的反序列化器...

2. 另一种选择

错误似乎是由于 PropDeser.deserialize() 方法消耗了父级的闭合标签。

java.lang.NullPointerException
at com.fasterxml.jackson.databind.deser.impl.BeanPropertyMap.find(BeanPropertyMap.java:160)
at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:287)
at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:112)
at com.fasterxml.jackson.databind.deser.std.CollectionDeserializer.deserialize(CollectionDeserializer.java:226)
at com.fasterxml.jackson.databind.deser.std.CollectionDeserializer.deserialize(CollectionDeserializer.java:203)
at com.fasterxml.jackson.databind.deser.std.CollectionDeserializer.deserialize(CollectionDeserializer.java:23)
at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:2575)
at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:1766)
at my.test.Test.deserializerTest(Test.java:57)

XML 流 中是否有可能进行向后迭代?方法如何知道何时停止?我一点头绪也没有。

1个回答

21

使用Jackson XML模块2.1的@JacksonXmlElementWrapper(useWrapping=false)注解,应该可以处理“未包装”的列表元素样式。

结构应该类似于:

@JacksonXmlRootElement(localName="parents")
public class Parents {
  @JacksonXmlElementWrapper(useWrapping=false)
  public List<Parent> parent;
}

public class Parent {
  @JacksonXmlProperty(isAttribute=true)
  public String name;

  public String description;

  @JacksonXmlElementWrapper(useWrapping=false)
  public List<Prop> prop;
}

public class Prop {
  @JacksonXmlProperty(isAttribute=true)
  public String name;

  public String value;
}

你的解决方案非常接近。

请注意,如果使用内部类,则需要在声明中添加“static”关键字。 我已经测试了2.1.4版本,并且它对我有效。


有没有办法在映射器上将useWrapping=false设置为默认值,这样我们就不需要为每个(第三方)对象进行注释设置了? - dan carter
是的,有这样的设置;在JacksonXmlModule中有一个名为setDefaultUseWrapper的方法。 - StaxMan
val xmlModule = JacksonXmlModule()xmlModule.setDefaultUseWrapper(false)val xmlMapper = XmlMapper(xmlModule) - moodboom

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接