这是我的情景。我有一个通用类:
public class Tuple<T> extends ArrayList<T> {
//...
public Tuple(T ...members) {
this(Arrays.asList(members));
}
@XmlElementWrapper(name = "tuple")
@XmlElement(name = "value")
public List<T> getList() {
return this;
}
}
和一个儿童班:
public class StringTuple extends Tuple<String> {
public StringTuple(String ...members) {
super(members);
}
//explanation of why overriding this method soon ...
@XmlElementWrapper(name = "tuple")
@XmlElement(name = "value")
@Override
public List<String> getList() {
return this;
}
}
这里引用了这些类:
@XmlRootElement(namespace = "iv4e.xml.jaxb.model")
public class Relation {
private Tuple<StringTuple> relationVars;
//...
@XmlElementWrapper(name = "allRelationVars")
@XmlElement(name = "relationVarsList")
public Tuple<StringTuple> getRelationVars() {
return relationVars;
}
}
然后使用以下内容创建Relation对象:
Relation rel = new Relation();
rel.setRelationVars(new Tuple<StringTuple>(
new StringTuple("RelationshipVar1"), new StringTuple("RelationshipVar2")));
编组此对象后,Xml输出如下:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<ns2:relation xmlns:ns2="iv4e.xml.jaxb.model" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="">
<allRelationVars>
<relationVarsList>
<tuple>
<value xmlns:xs="http://www.w3.org/2001/XMLSchema" xsi:type="xs:string">RelationshipVar1</value>
</tuple>
<tuple>
<value>RelationshipVar1</value>
</tuple>
</relationVarsList>
<relationVarsList>
<tuple>
<value xmlns:xs="http://www.w3.org/2001/XMLSchema" xsi:type="xs:string">RelationshipVar2</value>
</tuple>
<tuple>
<value>RelationshipVar2</value>
</tuple>
</relationVarsList>
</allRelationVars>
</ns2:relation>
所以value
元素是重复的!
现在,类StringTuple用List<T> getList()
覆盖List<String> getList()
的原因是避免列表中每个成员(xml中的xmlns:xs
元素)生成令人烦恼的value
属性文献)。
但是然后列表中的每个成员在输出中显示两次。显然,这是因为重写的父方法和子方法都使用@XmlElement
进行了注释。
所以我的主要问题是:有一种方法可以忽略在Jaxb中用@XmlElement
注释的重写方法? (考虑到覆盖方法也用@XmlElement
注释)
我发现一篇旧帖子报告了类似的问题:http://old.nabble.com/@XmlElement-on-overridden-methods-td19101616.html,但我还没有找到任何解决方案。
另请注意,在父类(@XmlTransient
)的getList
方法中添加Tuple<T>
注释可以解决此问题但会生成其他问题,因为父类不是抽象的并且是单独使用的在其他情况下。
一方面的第二个问题:是否可以在根节点声明xmlns:xs
属性而不是它 - 在需要它的每个节点中出现?我知道这可以用NamespacePrefixMapper
类完成,但由于它是一个非标准的SUN内部类,我宁愿使用更多独立于实现的方法。
提前感谢您的任何反馈!
答案 0 :(得分:3)
您可以使用以下方法在父项上标记属性@XmlTransient
,在子项上标记@XmlElement
:
<强>父强>
package forum7851052;
import java.util.ArrayList;
import java.util.List;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.XmlTransient;
@XmlRootElement
public class Parent<T> {
private List<T> item = new ArrayList<T>();
@XmlTransient
public List<T> getItem() {
return item;
}
public void setItem(List<T> item) {
this.item = item;
}
}
<强> IntegerChild 强>
package forum7851052;
import java.util.List;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class IntegerChild extends Parent<Integer> {
@Override
@XmlElement
public List<Integer> getItem() {
return super.getItem();
}
@Override
public void setItem(List<Integer> item) {
super.setItem(item);
}
}
<强> StringChild 强>
package forum7851052;
import java.util.List;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class StringChild extends Parent<String> {
@Override
@XmlElement
public List<String> getItem() {
return super.getItem();
}
@Override
public void setItem(List<String> item) {
super.setItem(item);
}
}
<强>演示强>
package forum7851052;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(Parent.class, IntegerChild.class, StringChild.class);
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
IntegerChild integerChild = new IntegerChild();
integerChild.getItem().add(1);
integerChild.getItem().add(2);
marshaller.marshal(integerChild, System.out);
StringChild stringChild = new StringChild();
stringChild.getItem().add("A");
stringChild.getItem().add("B");
marshaller.marshal(stringChild, System.out);
}
}
<强>输出强>
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<integerChild>
<item>1</item>
<item>2</item>
</integerChild>
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<stringChild>
<item>A</item>
<item>B</item>
</stringChild>
答案 1 :(得分:2)
这可能很旧,但它是搜索“JAXB重复字段”时的第一个结果
偶然发现了同样的问题,这对我有用了:
@XmlRootElement
@XmlAccessorType(XmlAccessType.NONE) // <-- made the difference
public abstract class ParentClass
{
...
}
@XmlRootElement
public class ChildClass extends ParentClass
{
...
}