我有一个深层的XML结构,有许多无意义的包装器,我正在映射到单个Java类。有几个不同的文件,其内容和结构略有不同。因为我希望能够将结果类转换为易于比较的东西(例如,每个表示包含一个名称),我想知道是否可以为@XmlPath
指定通配符。
继承是我第一次想到(部分)解决问题,但由于结构顶部有一个包装元素不同,我不知道如何解决。
示例XML结构
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<s:root xsi:schemaLocation="http://www.example.eu/test ResourceSchema.xsd" xmlns:s="http://www.example.eu/test" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<s:resource|s:data|s:container|s:stackoverflow>
<s:information>
<s:date>2013-07-04</s:date>
<s:name>This example does not work</s:name>
</s:information>
<s:elements>
<s:refobj>
<s:id>1</s:id>
<s:source>First Source</s:source>
</s:refobj>
<s:refobj>
<s:id>2</s:id>
<s:source>Second Source</s:source>
</s:refobj>
<s:refobj>
<s:id>5</s:id>
<s:source>Fifth Source</s:source>
</s:refobj>
</s:elements>
</s:resource|s:data|s:container|s:stackoverflow>
</s:root>
XML中的第二个元素显然是无效的,尽管结构不同并且几乎可以包含任何内容。然而,每个单元中都存在第三级元素information
结构,它甚至包含有关XML中表示的元素类型的信息。
一个快速的解决方案是为每个可能的元素创建一个类,然后尝试/捕获所有元素直到成功,尽管这似乎是一个可怕的解决方案。
解决像这样的XML相关问题的正确方法是什么?我没有可能更改结构,我无法访问架构告诉我哪些元素有多个名称。
答案 0 :(得分:2)
MOXy目前不支持@XmlPath
中的通配符,但您可以使用StreamReaderDelegate
来完成此操作。
DOMAIN MODEL
的根强> 的
package forum17527941;
import java.util.Date;
import javax.xml.bind.annotation.*;
import org.eclipse.persistence.oxm.annotations.*;
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Root {
@XmlPath("s:stackoverflow/s:information/s:date/text()")
@XmlSchemaType(name="date")
private Date date = new Date();
@XmlPath("s:stackoverflow/s:information/s:name/text()")
private String name;
}
的包信息强> 的
@XmlSchema(
namespace="http://www.example.eu/test",
xmlns={
@XmlNs(prefix="s", namespaceURI="http://www.example.eu/test")
},
elementFormDefault=XmlNsForm.QUALIFIED)
package forum17527941;
import javax.xml.bind.annotation.*;
DEMO CODE
的演示强> 的
package forum17527941;
import javax.xml.bind.*;
import javax.xml.stream.*;
import javax.xml.stream.util.StreamReaderDelegate;
import javax.xml.transform.stream.StreamSource;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(Root.class);
XMLInputFactory xif = XMLInputFactory.newFactory();
StreamSource xml = new StreamSource("src/forum17527941/input.xml");
XMLStreamReader xsr = xif.createXMLStreamReader(xml);
xsr = new StreamReaderDelegate(xsr) {
@Override
public String getLocalName() {
String localName = super.getLocalName();
if("resource".equals(localName) || "data".equals(localName) || "container".equals(localName)) {
return "stackoverflow";
}
return localName;
}
};
Unmarshaller unmarshaller = jc.createUnmarshaller();
Root root = (Root) unmarshaller.unmarshal(xsr);
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.setProperty(Marshaller.JAXB_SCHEMA_LOCATION, "http://www.example.eu/test ResourceSchema.xsd");
marshaller.marshal(root, System.out);
}
}
的 input.xml中强> 的
通过以下输入,StreamReaderDelegate
将导致resource
元素报告为匹配的stackoverflow
。
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<s:root xsi:schemaLocation="http://www.example.eu/test ResourceSchema.xsd" xmlns:s="http://www.example.eu/test" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<s:resource>
<s:information>
<s:date>2013-07-04</s:date>
<s:name>This example does not work</s:name>
</s:information>
</s:resource>
</s:root>
的输出强> 的
输出将匹配域模型中的映射。
<?xml version="1.0" encoding="UTF-8"?>
<s:root xsi:schemaLocation="http://www.example.eu/test ResourceSchema.xsd" xmlns:s="http://www.example.eu/test" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<s:stackoverflow>
<s:information>
<s:date>2013-07-04</s:date>
<s:name>This example does not work</s:name>
</s:information>
</s:stackoverflow>
</s:root>
更多信息