JAX-WS(JAXB)和unmarshalling混合xs:anyType

时间:2016-04-27 22:16:13

标签: java xsd jaxb jax-ws

我有一个网络服务,我必须使用xcj(针对.xsd)和wsimport(针对.wsdl)定义生成的代理来调用。它是外部的(即无法编辑),巨大且不断变化(无法手动调整生成的类)。

它有一个类似超类xs:anyType元素mixed="true",类似于下面的示例。

<Foo type="Bar">
    <id>123</id>
</Foo>

<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
    <xs:element name="Foo">
        <xs:complexType>
            <xs:complexContent mixed="true">
                <xs:extension base="xs:anyType">
                    <xs:attribute name="type" type="xs:string"/>
                </xs:extension>
            </xs:complexContent>
        </xs:complexType>
    </xs:element>

    <xs:element name="Bar">
        <xs:complexType>
            <xs:sequence>
                <xs:element name="id" type="xs:int"/>
                <xs:element name="name" type="xs:string"/>
            </xs:sequence>
        </xs:complexType>
    </xs:element>
</xs:schema>

xjc / wsimport生成Foo,如下所示

@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "", propOrder = { "content" })
@XmlRootElement(name = "Foo")
public class Foo {
    @XmlMixed
    @XmlAnyElement
    protected List<Object> content;
    @XmlAttribute(name = "type")
    protected String type;
    @XmlAnyAttribute
    private Map<QName, String> otherAttributes = new HashMap<QName, String>();

我可以使用Document.getChildNodes()将Foo对象编组为 root-less 表单(虽然我不介意找到更简洁的方法),但我无法弄清楚如何解组响应的content列表(代理完成后ArrayList<ElementNSImpl>)到强类型Bar

public static void main(String[] args) throws Exception {
    Foo foo = new Foo();
    foo.type = "Bar";

    Bar bar = new Bar();
    bar.id = 123;

    JAXBContext context = JAXBContext.newInstance(Foo.class, Bar.class);
    Marshaller marshaller = context.createMarshaller();
    marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

    Document document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
    marshaller.marshal(bar, document);

    NodeList nodes = document.getFirstChild().getChildNodes();
    foo.content = IntStream.range(0, nodes.getLength()).mapToObj(nodes::item).collect(Collectors.toList());

    StringWriter writer = new StringWriter();
    marshaller.marshal(foo, writer);
    System.out.println(writer);

    Unmarshaller unmarshaller = context.createUnmarshaller();

    // ???
    unmarshaller.unmarshal(foo.content, Bar.class);
}

1 个答案:

答案 0 :(得分:0)

我想我解决了。尽管如此,还是向任何人提出了一个更清洁的解决方案。

public static List<Object> marshal(Object value) {
    try {
        Class<?> type = value.getClass();
        if (type.isAnonymousClass())
            type = type.getSuperclass();

        Document document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
        Marshaller marshaller = JAXBContext.newInstance(type).createMarshaller();
        marshaller.marshal(new JAXBElement<>(QName.valueOf("root"), Object.class, value), document);

        NodeList nodes = document.getDocumentElement().getChildNodes();
        return IntStream.range(0, nodes.getLength()).mapToObj(nodes::item).collect(Collectors.toList());
    } catch (Exception e) {
        throw new RuntimeException(e);
    }
}

public static <T> T unmarshal(List<Object> content, Map<QName, String> attributes, Class<T> type) {
    try {
        Document document = DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
        document.appendChild(document.createElement("root"));

        if (attributes != null)
            attributes.forEach((q, s) -> document.getDocumentElement().setAttribute(q.getLocalPart(), s));

        if (content != null)
            content.forEach(o -> document.getDocumentElement().appendChild(document.importNode((Node) o, true)));

        Unmarshaller unmarshaller = JAXBContext.newInstance(type).createUnmarshaller();
        return unmarshaller.unmarshal(document, type).getValue();
    } catch (Exception e) {
        throw new RuntimeException(e);
    }
}