我有一个org.w3c.dom.Element,我将从自己的XmlAdapter返回一个自定义的@XmlElement,我想将它作为JAXB对象的一部分作为任意XML包含在内(我知道我必须手工制作XSD)。但是,JAXB提示

org.w3c.dom.Element is an interface, and JAXB can't handle interfaces.

显然,w3c XML类型是not supported as Java types,这很遗憾。但是,除此之外,当使用明显受支持的javax.xml.transform.Result时,我会遇到相同的错误。

如何在JAXB中包括任意XML元素作为元素?

注意:按照https://forums.oracle.com/thread/1668210,我也尝试过
MessageFactory factory = MessageFactory.newInstance();
message = factory.createMessage();
SOAPElement element = message.getSOAPBody().addDocument(doc);

但这也给了同样的错误。

最佳答案

TL; DR

只要将值类型指定为XmlAdapter(不是org.w3c.dom.Element),就可以使用Object将域对象转换为Element实例。

下面是一个完整的示例。

XmlAdapter

类型为java.lang.Object的字段/属性会将未知内容保留为DOM节点。您可以在用例中通过将XmlAdapter中的值类型指定为Object来利用它。您将需要确保从marshal方法返回的根元素与@XmlElement批注定义的字段/属性匹配。

import javax.xml.bind.annotation.adapters.XmlAdapter;
import javax.xml.parsers.*;
import org.w3c.dom.*;

public class BarAdapter extends XmlAdapter<Object, Bar>{

    private DocumentBuilder documentBuilder;

    public BarAdapter() {
        try {
            DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
            documentBuilder = dbf.newDocumentBuilder();
        } catch(Exception e) {
            // TODO - Handle Exception
        }
    }

    @Override
    public Bar unmarshal(Object v) throws Exception {
        Bar bar = new Bar();
        Element element = (Element) v;
        bar.value = element.getTextContent();
        return bar;
    }

    @Override
    public Object marshal(Bar v) throws Exception {
        Document document = documentBuilder.newDocument();
        Element root = document.createElement("bar");
        root.setTextContent(v.value);
        return root;
    }

}

Java模型

Foo
@XmlJavaTypeAdapter批注用于引用XmlAdapter
import javax.xml.bind.annotation.*;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Foo {

    @XmlJavaTypeAdapter(BarAdapter.class)
    private Bar bar;

}

酒吧
import javax.xml.bind.annotation.*;

@XmlAccessorType(XmlAccessType.FIELD)
public class Bar {

    String value;

}

示范代码

演示

由于创建DocumentBuilderFactory会产生成本,因此我们可以通过在Marshaller上设置实例来利用JAXB处理XmlAdapter的有状态实例的功能。
import java.io.File;
import javax.xml.bind.*;

public class Demo {

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

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        File xml = new File("src/forum18272059/input.xml");
        Foo foo = (Foo) unmarshaller.unmarshal(xml);

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setAdapter(new BarAdapter());
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(foo, System.out);
    }

}

input.xml/输出
<?xml version="1.0" encoding="UTF-8"?>
<foo>
    <bar>Hello World</bar>
</foo>

10-07 19:46