要了解如何使用@XmlAnyElement
,我创建了以下测试服务:
@WebService(serviceName = "TestServices")
@Stateless()
public class TestServices {
@WebMethod(operationName = "testMethod")
public ServiceResult testMethod() {
ServiceResult result = new ServiceResult();
result.addObject(new SimpleObj(1, 2));
result.addObject(new SimpleObj(3, 4));
return result;
}
}
SimpleObj
是一个包含2个int
字段的简单类。以下是ServiceResult
类的代码:
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
@XmlSeeAlso({SimpleObj.class})
public class ServiceResult {
@XmlAnyElement(lax = true)
private List<Object> body;
public void addObject(Object objToAdd) {
if (this.body == null)
this.body = new ArrayList();
this.body.add(objToAdd);
}
// Getters and Setters
}
为了使用上述服务,我使用以下Main
类创建了一个appclient:
public class Main {
@WebServiceRef(wsdlLocation = "META-INF/wsdl/localhost_8080/TestServices/TestServices.wsdl")
private static TestServices_Service service;
private static TestServices port;
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
port = service.getAdminServicesPort();
ServiceResult result = port.testMethod();
for (Object o : result.getAny()) {
System.out.println("TEST: " + o);
}
}
}
根据文档,使用@XmlAnyElement
,unmarshaller会急切地将此元素解组为JAXB对象。但是,我观察到的是JAXB只将我的对象解析为JAXBElement
而不是一直进入SimpleObj
。
如果您能告诉我如何从SimpleObj
中获取ServiceResult
,我将非常感激。
更新 :
以下是SimpleObj
类:
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class SimpleObj {
private int a;
private int b;
public SimpleObj() {}
public SimpleObj(int a, int b) {
this.a = a;
this.b = b;
}
// Getters and Setters
}
答案 0 :(得分:1)
我无法重现您所看到的问题。下面是一些与JAXB直接交互的演示代码。
import java.io.*;
import javax.xml.bind.*;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(ServiceResult.class);
Unmarshaller unmarshaller = jc.createUnmarshaller();
StringReader xml = new StringReader("<serviceResult><simpleObj/><simpleObj/></serviceResult>");
ServiceResult result = (ServiceResult) unmarshaller.unmarshal(xml);
for(Object item : result.getBody()) {
System.out.println(item.getClass());
}
}
}
运行演示代码的输出显示,在SimpleObj
注释的字段中是@XmlAnyElement(lax=true)
的实例。
class forum27871349.SimpleObj
class forum27871349.SimpleObj
在旁注中,我已经阅读了关于@XmlAnyElement和的博客文章 我从来没有见过你必须包含@XmlSeeAlso({SimpleObj.class}) 你的任何一个例子。
我不确定为什么我从未在我的示例中使用@XmlSeeAlso
。
然而,在我的情况下,如果我没有这个,我会有错误 在这种情况下,可以说Class ***或者它的任何超类。 如果你能告诉我是否有办法让所有人成功,那就太棒了 不使用@XmlSeeAlso
的消费者已知的这些类
当您自己创建JAXBContext
时,您只需要在@XmlSeeAlso
注释中包含您引用的任何内容,作为用于引导JAXBContext
的类的一部分。< / p>
JAXBContext jc = JAXBContext.newInstance(ServiceResult.class, SimpleObj.class);
在您无法直接访问JAXBContext
的JAX-WS(或JAX-RS)设置中,我建议您使用@XmlSeeAlso
注释,就像您所做的那样。
关于@XmlAnyElement,从文档中,我想如果 unmarshaller无法将元素解组为JAXB对象或 JAXBElement对象,我至少会得到一个DOM节点。
当您使用@XmlAnyElement(lax=true)
映射属性时,将发生以下情况:
@XmlRootElement
,那么您将获得该类的实例。@XmlElementDecl
上的类的ObjectFactory
或使用@XmlRegistry
注释的另一个类,那么您将获得包含在{{的实例中的该类的实例1}}。JAXBElement
。我将在下面举例说明。
<强>的ObjectFactory 强>
Element
<强>演示强>
import javax.xml.namespace.QName;
import javax.xml.bind.JAXBElement;
import javax.xml.bind.annotation.*;
@XmlRegistry
public class ObjectFactory {
@XmlElementDecl(name="simpleObjJAXBElement")
public JAXBElement<SimpleObj> createSimpleObj(SimpleObj simpleObj) {
return new JAXBElement<SimpleObj>(new QName("simpleObjJAXBElement"), SimpleObj.class, simpleObj);
}
}
<强>输出强>
import java.io.*;
import javax.xml.bind.*;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(ServiceResult.class, ObjectFactory.class);
Unmarshaller unmarshaller = jc.createUnmarshaller();
StringReader xml = new StringReader("<serviceResult><simpleObj/><unmapped/><simpleObjJAXBElement/></serviceResult>");
ServiceResult result = (ServiceResult) unmarshaller.unmarshal(xml);
for(Object item : result.getBody()) {
System.out.println(item.getClass());
}
}
}