9

问题

给定以下 XML 配置文件:

<main>
  <name>JET</name>
  <maxInstances>5</maxInstances>
  <parameters>
    <a>1</a>
    <b>
      <b1>test1</b1>
      <b2>test2</b2>
    </b>
  </parameters>
</main>

我需要提取 name 和 maxInstances 元素的值,然后提取参数元素的整个内部文本。例如

name = "JET"
maxInstances = 5
parameters = "<a>1</a><b><b1>test1</b1><b2>test2</b2></b>"

最终,参数块可以包含任何格式良好的 XML。

尝试的解决方案

以下代码适用于 name 和 maxInstances 但不适用于参数:

@XmlRootElement(name="main")
public class Main {

    @XmlElement(name="name", required="true")
    private String name;

    @XmlElement(name="maxInstances", required="true")
    private Integer maxInstances;

    @XmlElement(name="parameters")
    private String parameters;

}

我尝试根据以下想法查看解决方案,但找不到合适的解决方案。

是否有不同的类型可以用于表示 XML 树的参数对象,我可以解析以生成字符串?例如

@XmlElement(name="parameters")
private XmlNodeObject parametersNode;

public String getParameters() {
    // Collapse node to single line of text
    return innerText;
}

还是我需要使用一些不同类型的注释?

@XmlSpecialAnnotation(...)
@XmlElement(name="parameters")
private String parameters;

我需要切换到不同风格的解析器吗?使用两种风格的解析器是好还是坏?

4

2 回答 2

12

您可以使用bmargulies 所描述的@XmlAnyElement注释。要映射到您问题中的对象模型,您可以利用.DOMHandler

主要的

import javax.xml.bind.annotation.*;

@XmlRootElement(name="main")
@XmlAccessorType(XmlAccessType.FIELD)
public class Main {

    private String name;

    private Integer maxInstances;

    @XmlAnyElement(value=ParameterHandler.class)
    private String parameters;

}

参数处理程序

import java.io.*;
import javax.xml.bind.ValidationEventHandler;
import javax.xml.bind.annotation.DomHandler;
import javax.xml.transform.Source;
import javax.xml.transform.stream.*;

public class ParameterHandler implements DomHandler<String, StreamResult> {

    private static final String PARAMETERS_START_TAG = "<parameters>";
    private static final String PARAMETERS_END_TAG = "</parameters>";
    private StringWriter xmlWriter = new StringWriter(); 

    public StreamResult createUnmarshaller(ValidationEventHandler errorHandler) {
        return new StreamResult(xmlWriter);
    }

    public String getElement(StreamResult rt) {
        String xml = rt.getWriter().toString();
        int beginIndex = xml.indexOf(PARAMETERS_START_TAG) + PARAMETERS_START_TAG.length();
        int endIndex = xml.indexOf(PARAMETERS_END_TAG);
        return xml.substring(beginIndex, endIndex);
    }

    public Source marshal(String n, ValidationEventHandler errorHandler) {
        try {
            String xml = PARAMETERS_START_TAG + n.trim() + PARAMETERS_END_TAG;
            StringReader xmlReader = new StringReader(xml);
            return new StreamSource(xmlReader);
        } catch(Exception e) {
            throw new RuntimeException(e);
        }
    }

}

演示

import java.io.File;
import javax.xml.bind.*;

public class Demo {

    public static void main(String[] args) throws Exception  {
        JAXBContext jc = JAXBContext.newInstance(Main.class);

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        Main main = (Main) unmarshaller.unmarshal(new File("input.xml"));

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(main, System.out);
    }

}
于 2011-04-05T14:38:51.570 回答
6

最接近的方法是将“参数”映射到 DOM 树,方法是将变量声明为org.w3c.dom.Node. (实际上,声明一个 JAXBElement)。

有关详细信息,请参阅http://jaxb.java.net/guide/Avoid_strong_databinding.html。这为您提供了模式优先的处方,您可以通过 xsd2java 运行该模式并查看输出来了解如何从 java 开始。

要获取字符串,您必须从 DOM 序列化。

或者,更具体地说:

这个页面在这里描述 xsd:any 处理,因此

  @XmlAnyElement
  public List<Element> getParameters();

ElementDOM 接口在哪里。

于 2011-04-04T11:26:49.013 回答