我有一个定义以下类型的模式:
<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:任何使用过的JAXB?似乎没有什么可以工作,因为JAXB将Payload转换为Object,并且它不会仅对Object中的任何内容进行序列化.这一切都在Axis2内部,因此达到这一点非常具有挑战性.
解决方法:
下面我将演示JAXB (JSR-222)和任何一个例子:
有效载荷
any属性使用@XmlAnyElement注释(lax = true).这意味着对于该属性,如果元素通过@XmlRootElement或@XmlElementDecl与类关联,则相应对象的实例将用于填充属性,否则该元素将被设置为org.w3c.dom的实例.元件.
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注释的类的示例.在这个用例中,我们将在使用@XmlRegistry注释的工厂类(通常称为ObjectFactory)上利用@XmlElementDecl注释.
package forum13941747;
public class Bar {
}
的ObjectFactory
下面是为Bar类指定@XmlElementDecl注释的示例.
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);
}
}
input.xml中
下面是我们将用于此示例的输入文档.有3个元素对应于any属性.第一个对应于Foo类上的@XmlRootElement注释.第二个对应于Bar类的@XmlElementDecl注释,第三个对应于任何域类.
<?xml version="1.0" encoding="UTF-8"?>
<payload>
<foo/>
<bar/>
<other/>
</payload>
演示
在下面的演示代码中,我们将解组输入文档,然后在生成的any属性中输出对象的类,然后将有效负载对象封送回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实例.另一个元素成为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>