12

我有一个定义以下类型的架构:

<xsd:complexType name="Payload">
   <xsd:sequence>
      <xsd:any namespace="##any" minOccurs="0" maxOccurs="unbounded" processContents="lax"/>
   </xsd:sequence>
</xsd:complexType>

这会创建一个像这样的对象:

@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "Payload", propOrder = {
    "any"
})
public class Payload {

    @XmlAnyElement(lax = true)
    protected List<Object> any;
}

现在我尝试将另一个生成的 JAXB 对象添加到该 Payload,执行如下操作:

Class payloadClass = ...;
JAXBContext context = JAXBContext.newInstance( WrapperRequest.class, payloadClass);
...
marshaller.marshal( wrappedRequest );

但是我得到了一个可怕的异常,看起来它永远不会起作用,所以我决定先将有效负载对象序列化为 XML,然后将其作为字符串添加到有效负载中。

StringWriter writer = new StringWriter();
JAXBContext context = JAXBContext.newInstance( sdoRequest.getClass() );
Marshaller marshaller = context.createMarshaller();
marshaller.marshal(new JAXBElement(new QName("uri", sdoRequest.getClass().getSimpleName()), sdoRequest.getClass(), sdoRequest), writer);
payload.getAny().add( writer.toString() );

这会引发异常,说“java.lang.String”不包含@XmlRootElement。

那么 xs:any 将如何与 JAXB 一起使用呢?似乎什么都不想工作,因为 JAXB 将 Payload 转换为 Object,并且它不会序列化 Object 中的任何内容。这也都在 Axis2 内部,因此达到这一点非常具有挑战性。

4

3 回答 3

18

下面我将演示JAXB (JSR-222)any举例:

有效载荷

any属性用 注释@XmlAnyElement(lax=true)。这意味着对于该属性,如果元素通过@XmlRootElement或与类关联,@XmlElementDecl则将使用相应对象的实例来填充该属性,否则该元素将被设置为org.w3c.dom.Element.

package forum13941747;

import java.util.List;
import javax.xml.bind.annotation.*;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "Payload", propOrder = {
    "any"
})
public class Payload {

    @XmlAnyElement(lax = true)
    protected List<Object> any;

}

下面是一个用 注释的类的例子@XmlRootElement

package forum13941747;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Foo {

}

酒吧

下面是一个没有@XmlRootElement注释的类的例子。在这个用例中,我们将利用用@XmlElementDecl注释的工厂类(通常称为ObjectFactory)上的注释@XmlRegistry

package forum13941747;

public class Bar {

}

对象工厂

下面是为类指定@XmlElementDecl注释的示例Bar

package forum13941747;

import javax.xml.bind.JAXBElement;
import javax.xml.bind.annotation.*;
import javax.xml.namespace.QName;

@XmlRegistry
public class ObjectFactory {

    @XmlElementDecl(name="bar")
    public JAXBElement<Bar> createBar(Bar bar) {
        return new JAXBElement<Bar>(new QName("bar"), Bar.class, bar);
    }

}

输入.xml

下面是我们将用于此示例的输入文档。有 3 个元素对应于该any属性。第一个对应于类上的@XmlRootElement注解Foo。第二个对应于类的@XmlElementDecl注释,Bar第三个不对应于任何域类。

<?xml version="1.0" encoding="UTF-8"?>
<payload>
    <foo/>
    <bar/>
    <other/>
</payload>

演示

在下面的演示代码中,我们将解组输入文档,然后在结果any属性中输出对象的类,然后将payload对象编组回 XML。

package forum13941747;

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

public class Demo {

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

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        File xml = new File("src/forum13941747/input.xml");
        Payload payload = (Payload) unmarshaller.unmarshal(xml);

        for(Object o : payload.any) {
            System.out.println(o.getClass());
        }

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

}

输出

下面是运行演示代码的输出。请注意与属性中的对象对应的类any。该foo元素成为Foo该类的一个实例。bar元素变成了 的实例,它JAXBElement拥有 的实例Bar。该other元素成为 的一个实例org.w3c.dom.Element

class forum13941747.Foo
class javax.xml.bind.JAXBElement
class com.sun.org.apache.xerces.internal.dom.ElementNSImpl
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<payload>
    <foo/>
    <bar/>
    <other/>
</payload>
于 2012-12-19T11:09:26.603 回答
1

使用对象工厂对对象进行如下处理,您无需在 DemoType.java 中有 @XmlRootElement 。,

DemoType demoServiceRequest = new DemoType();
ObjectFactory obDemo = new ObjectFactory();  
Request requestObject = new Request();     
requestObject.setAny(obDemo.createDemo(demoServiceRequest));

并在 Request.java 中添加 DemoType 类,如 @XmlSeeAlso({DemoType.class})

于 2015-06-09T22:25:03.517 回答
0

如果您的有效负载是 XML 字符串,我设法使用以下代码解决了同样的问题:

import javax.xml.parsers.DocumentBuilderFactory;

//...

String XMLPAYLOAD = "...";

Payload payload = new ObjectFactory().createPayload();
try {
    payload.setAny(DocumentBuilderFactory
           .newInstance()
           .newDocumentBuilder()
           .parse(new InputSource(new StringReader(XMLPAYLOAD)))
           .getDocumentElement());
} catch (Exception e) {
    e.printStackTrace();
}

//...
于 2019-05-23T16:06:25.833 回答