我有一个 XML 文档(由 Adobe XFA 表单生成),其中包含如下数据:
<Position>
<PositionBorder>
<Title/>
<StartDate/>
<EndDate/>
</PositionBorder>
</Position>
由于该文件是在其他地方定义的,因此我无法随意更改所获得的 XML 的格式。
在我的 Java 代码中,我创建了一个 Position 类,其中包含标题、开始日期和结束日期。
我的问题是,当我使用 XStream 解析文件时,它需要一个 PositionBorder 类来保存标题和日期。我想基本上忽略边框并将所有字段放入 Position 类中。
我真正想做的是使用像convertAnother 方法这样的方法来转换position 元素的子元素。我尝试这样做,但失败了,因为我的 PositionConverter 被调用以获取 PositionBorder (当我调用 ConvertAnother 时)。
有人知道如何在解析时处理 XML 结构的崩溃吗?
使用自定义转换器并不是非常困难。这是一个有点长的示例,但我希望它足够简单,足以让您了解需要执行的操作的要点:
import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import com.thoughtworks.xstream.converters.Converter;
import com.thoughtworks.xstream.converters.MarshallingContext;
import com.thoughtworks.xstream.converters.UnmarshallingContext;
import com.thoughtworks.xstream.io.HierarchicalStreamReader;
import com.thoughtworks.xstream.io.HierarchicalStreamWriter;
public final class ConverterTest {
public static void main(String[] args) {
XStream xstream = new XStream();
xstream.autodetectAnnotations(true);
xstream.registerConverter(new PositionConverter());
final Position position = new Position();
position.setTitle("The Title");
position.setStartDate("The Start Date");
position.setEndDate("The End Date");
final String xml = xstream.toXML(position);
System.out.println("Generated XML:");
System.out.println(xml);
final Position genPosition = (Position) xstream.fromXML(xml);
System.out.println("Generated Position:");
System.out.println("\tTitle: " + genPosition.getTitle());
System.out.println("\tStart Date: " + genPosition.getStartDate());
System.out.println("\tEnd Date: " + genPosition.getEndDate());
}
@XStreamAlias("Position")
private static class Position {
public String getEndDate() {
return endDate;
}
public void setEndDate(String endDate) {
this.endDate = endDate;
}
public String getStartDate() {
return startDate;
}
public void setStartDate(String startDate) {
this.startDate = startDate;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
private String title;
private String startDate;
private String endDate;
}
private static class PositionConverter implements Converter {
public boolean canConvert(Class clazz) {
return Position.class == clazz;
}
public void marshal(Object value, HierarchicalStreamWriter writer, MarshallingContext context) {
Position position = (Position)value;
writer.startNode("PositionBorder");
writer.startNode("Title");
writer.setValue(position.getTitle());
writer.endNode();
writer.startNode("StartDate");
writer.setValue(position.getStartDate());
writer.endNode();
writer.startNode("EndDate");
writer.setValue(position.getEndDate());
writer.endNode();
writer.endNode();
}
public Object unmarshal(HierarchicalStreamReader reader, UnmarshallingContext context) {
Position position = new Position();
// move it to <PositionBorder> tag.
reader.moveDown();
// now move it to <Title> tag.
reader.moveDown();
String title = reader.getValue();
position.setTitle(title);
reader.moveUp(); // moves back to <PositionBorder>
reader.moveDown(); // should move down to <StartDate> tag
String startDate = reader.getValue();
position.setStartDate(startDate);
reader.moveUp(); // move back to <PositionBorder>
reader.moveDown(); // should move down to <EndDate> tag
String endDate = reader.getValue();
position.setEndDate(endDate);
reader.moveUp(); // move back to <PositionBorder>
return position;
}
}
}
尝试运行它,看看会发生什么。当然,您需要修改它以适合您自己的类型 - 我只是对 Position 的所有字段使用字符串(并且我确信您的 Position 类也不是嵌套的),但是从 String 转换约会(或其他什么)应该是相当微不足道的。
你需要关注的一件事(我可能没有明白)完全地在我的示例中)与您的 reader.moveDown() 和 reader.moveUp() 调用相匹配。 (而且,如果您要进行任何编组而不仅仅是解组——我不希望从您的问题中得到这一点——您也会希望匹配您的 writer.startNode() 和 writer.endNode() 调用.)这个例子可能不会造成任何问题,但我确信如果您正在做更大的事情或使用相同的 XStream 或 Converter 实例处理多个文件,它会引发问题。另外,如果您从无效位置尝试 reader.moveDown(),您将得到一个非常丑陋的异常 - 它应该非常明显。
我必须稍微尝试一下 moveUp/moveDown 方法才能将它们放在正确的位置,因此我确信您需要对其进行测试并进行调整,直到获得所需的内容。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)