我有一个 Animal 类和一个名为 AnimalExtension 的 Animal 扩展。
public class Animal
public class AnimalExtension extends Animal
这两个类之间的唯一区别是 AnimalExtension 有另一个名为 animalId 的实例变量。 Animal 没有这个实例变量。
我也有自己的数据类型,我想将其编码和取消编码为 XML。这种数据类型称为 AnimalList。在 AnimalList 中,有一个 Animals 列表作为实例变量。
@XmlType(name = "AnimalList")
public class AnimalList{
private List<Animal> animalList;
....
animalList 可以同时包含 Animal 和 AnimalExtension。但是,在 XML 上,我不想将元素命名为 AnimalExtension;我希望它们都具有 Animal 元素名称。我只希望在 JAXB 知道 Animal 实际上是 AnimalExtension 的一个实例时显示额外的属性。所以如果我有一个看起来像这样的列表
List<Animal> animalList = new LinkedList<Animal>();
AnimalExtension animalExtension = new AnimalExtension();
animalExtension.setAnimalId(1);
amimalExtension.setName("Don");
Animal animal = new Animal();
animal.setName("Mike");
animalList.add(animalExtension);
animalList.add(animal);
我希望 XML 看起来像
<AnimalList>
<Animal name="Don" id="1" />
<Animal name="Mike" />
</AnimalList>
这是我想做的
@XmlElements(
{
@XmlElement(name = "Animal", type = Animal.class),
@XmlElement(name = "Animal", type = AnimalExtension.class)
}
)
public List<Animal> getEntries() {
return animalList;
}
代码可以编译,但是当我尝试运行我的服务器时。它给了我这个与正在发生的事情无关的奇怪错误(BeanCreationException)。我尝试使 XmlElement 的名称对于每种类型都不同并且可行,但这里的挑战是使名称相同。
org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'encryptionPayloadContentProvider'
最佳答案
要映射此用例,您可以利用以下 XmlAdapter:
动物适配器
由于 AnimalExtension 是 Animal 的超集,我们将使用它来生成/使用 XML。然后我们将利用 animalId 属性的值来确定是否将 Animal 或 AnimalExtension 的实例返回到 AnimalList。
import javax.xml.bind.annotation.adapters.XmlAdapter;
public class AnimalAdapter extends XmlAdapter<AnimalExtension, Animal> {
@Override
public Animal unmarshal(AnimalExtension animalExtension) throws Exception {
if(0 != animalExtension.getAnimalId()) {
return animalExtension;
}
Animal animal = new Animal();
animal.setName(animalExtension.getName());
return animal;
}
@Override
public AnimalExtension marshal(Animal animal) throws Exception {
if(animal.getClass() == AnimalExtension.class) {
return (AnimalExtension) animal;
}
AnimalExtension animalExtension = new AnimalExtension();
animalExtension.setName(animal.getName());
return animalExtension;
}
}
IdAdapter
如果 animalId 的值为 0,我们将需要第二个 XmlAdapter 来抑制它:
import javax.xml.bind.annotation.adapters.XmlAdapter;
public class IdAdapter extends XmlAdapter<String, Integer> {
@Override
public Integer unmarshal(String string) throws Exception {
return Integer.valueOf(string);
}
@Override
public String marshal(Integer integer) throws Exception {
if(integer == 0) {
return null;
}
return String.valueOf(integer);
}
}
您的模型类将被注释如下:
动物列表
import java.util.ArrayList;
import java.util.List;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
@XmlRootElement(name="AnimalList")
public class AnimalList {
private List<Animal> animalList = new ArrayList<Animal>();
@XmlElement(name="Animal")
@XmlJavaTypeAdapter(AnimalAdapter.class)
public List<Animal> getEntries() {
return animalList;
}
}
动物
import javax.xml.bind.annotation.XmlAttribute;
public class Animal {
private String name;
@XmlAttribute
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
AnimalExtension
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
public class AnimalExtension extends Animal {
private int animalId;
@XmlAttribute(name="id")
@XmlJavaTypeAdapter(IdAdapter.class)
public int getAnimalId() {
return animalId;
}
public void setAnimalId(int animalId) {
this.animalId = animalId;
}
}
演示代码
以下演示代码可用于演示此解决方案:
import java.io.File;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(AnimalList.class);
Unmarshaller unmarshaller = jc.createUnmarshaller();
File xml = new File("input.xml");
AnimalList animalList = (AnimalList) unmarshaller.unmarshal(xml);
for(Animal animal : animalList.getEntries()) {
System.out.println(animal.getClass());
}
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(animalList, System.out);
}
}
将产生以下输出:
class AnimalExtension
class Animal
<?xml version="1.0" encoding="UTF-8"?>
<AnimalList xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<Animal name="Don" id="1"/>
<Animal name="Mike"/>
</AnimalList>
相关信息
您可能会发现以下信息很有用:
关于java - JAXB @XmlElements,不同类型但名称相同?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5007516/