我在两个名称空间中有四个类,两个在一个包中用作另一个包中的两个类的基类,例如:
@XmlAccessorType(XmlAccessType.FIELD)
public class Foo<T extends Bar> {
@XmlElement(name="Bar")
private List<T> bar;
}
和
@XmlAccessorType(XmlAccessType.FIELD)
public class Bar {
// something
}
派生类看起来像:
@XmlType(name="Foo" namespace="urn:foo/Bar")
@XmlRootElement(name="Foo")
public class DerivedFoo extends Foo<DerivedBar> {
// something
}
和
@XmlType(name="Bar" namespace="urn:foo/Bar")
public class DerivedBar extends Bar {
// something
}
在包含基类的包中,没有package-info.java
或包含命名空间信息的任何内容。包含派生类的包具有所有命名空间信息。
输出XML是DerivedFoo
类的命名空间,但正在将DerivedBar
类分配给未命名的命名空间,基本上是:
<Response xmlns:ns2="urn:foo/Bar">
<ns2:Foo>
<Bar>
</Bar>
</ns2:Foo>
</Response>
答案 0 :(得分:2)
我认为您正在寻找的是@XmlElementRef
和@XmlRootElement
的组合。使用@XmlElementRef
注释就像是说“这是一个东西;去看看它的@XmlRootElement
注释,找出如何序列化它。”
以下设置似乎可以满足您的需求。基类:
package base;
@XmlAccessorType(XmlAccessType.FIELD)
public class Foo<T extends Bar> {
@XmlElementRef
public List<T> bar;
}
package base;
@XmlAccessorType(XmlAccessType.FIELD)
public class Bar {
// something
}
package base;
@XmlRootElement(name = "Request")
@XmlAccessorType(XmlAccessType.FIELD)
public class Request {
@XmlElementRef
public Foo<?> foo;
}
和子类:
package derived;
@XmlRootElement(name = "Foo", namespace = "urn:foo/Bar")
public class DerivedFoo extends Foo<DerivedBar> {
// something
}
package derived;
@XmlRootElement(name = "Bar", namespace = "urn:foo/Bar")
public class DerivedBar extends Bar {
// something
}
从我的测试中看起来很不错:
@Test
public void tryIt() throws Exception {
JAXBContext context = JAXBContext.newInstance(Request.class, DerivedFoo.class, DerivedBar.class);
Marshaller marshaller = context.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
Request request = new Request();
DerivedFoo foo = new DerivedFoo();
request.foo = foo;
foo.bar = new ArrayList<DerivedBar>();
foo.bar.add(new DerivedBar());
marshaller.marshal(request, System.out);
}
输出:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<Request xmlns:ns2="urn:foo/Bar">
<ns2:Foo>
<ns2:Bar/>
</ns2:Foo>
</Request>
由于@XmlElementRef
在运行时动态查找XML绑定,如果您使用不同的命名空间声明另一个Bar
的子类,它将“正常工作”。 小心 但是,因为元素名称也是动态的,如果派生类并非都使用相同的name
属性,则可能会导致奇怪的结果:
<Request xmlns:ns2="urn:rock/roll" xmlns:ns3="urn:yummy">
<ns2:FooFighters>
<ns3:TikiBar/>
</ns2:FooFighters>
</Request>
嵌套类型可能并不太在意,但我敢打赌,消费Request
的人可能无法识别这些元素名称。