我在实现项目的Web服务层时正在处理一些循环引用。我正在使用jaxb(最新版本,2.2.7),甚至我查看了一些提示here和here我无法使用它。这是关于我的问题的基本SSCCE:
/*
* The service interface
*/
@WebService
public interface IMyWS {
@WebMethod
public List<Class1> cyclicTest();
}
/*
* Interface implementation
*/
@WebService(endpointInterface = "com.mycompany.ws.interfaces.IMyWS")
public class MyWS implements IMyWS {
@XmlRootElement
public static class Class1 {
@XmlTransient
private Class2 class2;
public Class1() {
}
public Class1(Class2 refClass) {
class2 = refClass;
}
public Class2 getClass2() {
return class2;
}
public void setClass2(Class2 class2) {
this.class2 = class2;
}
@Override
public String toString() {
return this.getClass().getSimpleName();
}
}
@XmlRootElement
public static class Class2 {
private Class1 class1;
public Class2() {
}
public Class1 getClass1() {
return class1;
}
public void setClass1(Class1 class1) {
this.class1 = class1;
}
@Override
public String toString() {
return this.getClass().getSimpleName();
}
}
@Override
public List<Class1> cyclicTest() {
//I create an instance of each class, having them a cyclic reference to the other instance
Class2 class2 = new Class2();
Class1 class1 = new Class1(class2);
class2.setClass1(class1);
return Arrays.asList(class1);
}
}
我在调用cyclicTest()
时实际处理的异常:
Caused by: javax.xml.bind.MarshalException
- with linked exception:
[com.sun.istack.SAXException2: Se ha detectado un ciclo en el gráfico de objeto. Esto provocará un XML con profundidad infinita: Class1 -> Class2 -> Class1]
at com.sun.xml.bind.v2.runtime.MarshallerImpl.write(MarshallerImpl.java:326)
at com.sun.xml.bind.v2.runtime.MarshallerImpl.marshal(MarshallerImpl.java:178)
at org.apache.cxf.jaxb.JAXBEncoderDecoder.writeObject(JAXBEncoderDecoder.java:537)
at org.apache.cxf.jaxb.JAXBEncoderDecoder.marshall(JAXBEncoderDecoder.java:233)
... 50 more
Caused by: com.sun.istack.SAXException2: Se ha detectado un ciclo en el gráfico de objeto. Esto provocará un XML con profundidad infinita: Class1 -> Class2 -> Class1
at com.sun.xml.bind.v2.runtime.XMLSerializer.reportError(XMLSerializer.java:249)
at com.sun.xml.bind.v2.runtime.XMLSerializer.pushObject(XMLSerializer.java:537)
at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsXsiType(XMLSerializer.java:631)
at com.sun.xml.bind.v2.runtime.property.SingleElementNodeProperty.serializeBody(SingleElementNodeProperty.java:158)
at com.sun.xml.bind.v2.runtime.ClassBeanInfoImpl.serializeBody(ClassBeanInfoImpl.java:361)
at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsXsiType(XMLSerializer.java:696)
at com.sun.xml.bind.v2.runtime.property.SingleElementNodeProperty.serializeBody(SingleElementNodeProperty.java:158)
at com.sun.xml.bind.v2.runtime.ClassBeanInfoImpl.serializeBody(ClassBeanInfoImpl.java:361)
at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsXsiType(XMLSerializer.java:696)
at com.sun.xml.bind.v2.runtime.property.ArrayElementNodeProperty.serializeItem(ArrayElementNodeProperty.java:69)
at com.sun.xml.bind.v2.runtime.property.ArrayElementProperty.serializeListBody(ArrayElementProperty.java:172)
at com.sun.xml.bind.v2.runtime.property.ArrayERProperty.serializeBody(ArrayERProperty.java:159)
at com.sun.xml.bind.v2.runtime.ClassBeanInfoImpl.serializeBody(ClassBeanInfoImpl.java:361)
at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsXsiType(XMLSerializer.java:696)
at com.sun.xml.bind.v2.runtime.ElementBeanInfoImpl$1.serializeBody(ElementBeanInfoImpl.java:156)
at com.sun.xml.bind.v2.runtime.ElementBeanInfoImpl$1.serializeBody(ElementBeanInfoImpl.java:131)
at com.sun.xml.bind.v2.runtime.ElementBeanInfoImpl.serializeBody(ElementBeanInfoImpl.java:333)
at com.sun.xml.bind.v2.runtime.ElementBeanInfoImpl.serializeRoot(ElementBeanInfoImpl.java:340)
at com.sun.xml.bind.v2.runtime.ElementBeanInfoImpl.serializeRoot(ElementBeanInfoImpl.java:76)
at com.sun.xml.bind.v2.runtime.XMLSerializer.childAsRoot(XMLSerializer.java:494)
at com.sun.xml.bind.v2.runtime.MarshallerImpl.write(MarshallerImpl.java:323)
... 53 more
我想我已经设置了正确的注释。我实际上缺少什么?
答案 0 :(得分:5)
使用@XmlID和@XmlIDREF可以解决您的问题。
这就是我的所作所为:
我在Class 1中添加了一个字符串字段作为id,并使用@XmlID
对其进行了注释。
然后,我使用setClass1()
注释了@XmlIDREF
方法。我的测试在下面的主要方法中。
/*
* Interface implementation
*/
@WebService(endpointInterface = "com.mycompany.ws.interfaces.IMyWS")
public class MyWS implements IMyWS {
@XmlRootElement
public static class Class1 {
@XmlID
private String id;
private Class2 class2;
public Class1() {
this.id = UUID.randomUUID().toString();
}
public Class1(Class2 refClass) {
this();
class2 = refClass;
}
public Class2 getClass2() {
return class2;
}
public void setClass2(Class2 class2) {
this.class2 = class2;
}
@Override
public String toString() {
return this.getClass().getSimpleName();
}
}
@XmlRootElement
public static class Class2 {
private Class1 class1;
public Class2() {
}
public Class1 getClass1() {
return class1;
}
@XmlIDREF
public void setClass1(Class1 class1) {
this.class1 = class1;
}
@Override
public String toString() {
return this.getClass().getSimpleName();
}
}
@Override
public List<Class1> cyclicTest() {
//I create an instance of each class, having them a cyclic reference to the other instance
Class2 class2 = new Class2();
Class1 class1 = new Class1(class2);
class2.setClass1(class1);
return Arrays.asList(class1);
}
public static void main(String[] args) throws JAXBException {
JAXBContext ctx = JAXBContext.newInstance(Class1.class);
Marshaller m = ctx.createMarshaller();
m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
List<Class1> class1s = new MyWS().cyclicTest();
for (Class1 c1 : class1s){
m.marshal(c1, System.out);
}
}
}
输出:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<class1>
<id>e9e53e73-9a96-4c7c-b919-3ed3d7aa4c5e</id>
<class2>
<class1>e9e53e73-9a96-4c7c-b919-3ed3d7aa4c5e</class1>
</class2>
</class1>
答案 1 :(得分:3)
当您在私有财产上添加@XmlTransient
时,您应该将XmlAccessType
(默认为XmlAccessType.PUBLIC_MEMBER
)更改为XmlAccessType.PROPERTY
PUBLIC_MEMBER
是JAXB中的默认访问类型。这意味着JAXB实现将为以下内容生成绑定:公共字段,带注释的字段,属性
从W A复制的代码:
@WebService(endpointInterface = "com.mycompany.ws.interfaces.IMyWS")
public class MyWS implements IMyWS {
@XmlRootElement
@XmlAccessorType(XmlAccessType.PROPERTY)
public static class Class1 {
private Class2 class2;
public Class1() {
}
public Class1(Class2 refClass) {
class2 = refClass;
}
@XmlTransient
public Class2 getClass2() {
return class2;
}
public void setClass2(Class2 class2) {
this.class2 = class2;
}
@Override
public String toString() {
return this.getClass().getSimpleName();
}
}
@XmlRootElement
public static class Class2 {
private Class1 class1;
public Class2() {
}
public Class1 getClass1() {
return class1;
}
public void setClass1(Class1 class1) {
this.class1 = class1;
}
@Override
public String toString() {
return this.getClass().getSimpleName();
}
}
public List<Class1> cyclicTest() {
//I create an instance of each class, having them a cyclic reference to the other instance
Class2 class2 = new Class2();
Class1 class1 = new Class1(class2);
class2.setClass1(class1);
return Arrays.asList(class1);
}
public static void main(String[] args) throws JAXBException {
JAXBContext ctx = JAXBContext.newInstance(Class1.class);
Marshaller m = ctx.createMarshaller();
m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
List<Class1> class1s = new MyWs().cyclicTest();
for (Class1 c1 : class1s){
m.marshal(c1, System.out);
}
}
答案 2 :(得分:0)
添加W Almir的答案,为了更简单地生成对象的唯一ID(无需向构造函数添加代码),请注释返回id的方法。例如:
@XmlID
public String getId()
{
return Integer.toString(System.identityHashCode(this));
}
这似乎不仅适用于将Java对象的循环图转换为XML,还适用于将XML转换为Java对象的循环图。