Кажется, JAXB не может прочитать то, что пишет. Рассмотрим следующий код:
interface IFoo {
void jump();
}
@XmlRootElement
class Bar implements IFoo {
@XmlElement
public String y;
public Bar() {
y = "";
}
public Bar(String y) {
this.y = y;
}
@Override
public void jump() {
System.out.println(y);
}
}
@XmlRootElement
class Baz implements IFoo {
@XmlElement
public int x;
public Baz() {
x = 0;
}
public Baz(int x) {
this.x = x;
}
@Override
public void jump() {
System.out.println(x);
}
}
@XmlRootElement
public class Holder {
private List<IFoo> things;
public Holder() {
things = new ArrayList<>();
}
@XmlElementWrapper
@XmlAnyElement
public List<IFoo> getThings() {
return things;
}
public void addThing(IFoo thing) {
things.add(thing);
}
}
// ...
try {
JAXBContext context = JAXBContext.newInstance(Holder.class, Bar.class, Baz.class);
Holder holder = new Holder();
holder.addThing(new Bar("1"));
holder.addThing(new Baz(2));
holder.addThing(new Baz(3));
for (IFoo thing : holder.getThings()) {
thing.jump();
}
StringWriter s = new StringWriter();
context.createMarshaller().marshal(holder, s);
String data = s.toString();
System.out.println(data);
StringReader t = new StringReader(data);
Holder holder2 = (Holder)context.createUnmarshaller().unmarshal(t);
for (IFoo thing : holder2.getThings()) {
thing.jump();
}
}
catch (Exception e) {
System.err.println(e.getMessage());
}
Конечно, это упрощенный пример. Дело в том, что мне приходится хранить в одной коллекции два совершенно по-разному реализованных класса, Bar и Baz. Ну, я заметил, что у них очень похожий публичный интерфейс, поэтому я создал интерфейс IFoo и сделал их двумя для его реализации. Теперь я хочу иметь инструменты для сохранения и загрузки этой коллекции в/из XML. К сожалению, этот код не совсем работает: коллекция сохраняется, но потом не загружается! Предполагаемый вывод:
1
2
3
some xml
1
2
3
. Но, к сожалению, фактический вывод:
1
2
3
some xml
com.sun.org.apache.xerces.internal.dom.ElementNSImpl cannot be cast to testapplication1.IFoo
. Видимо, мне нужно использовать аннотации по-другому? Или отказаться от JAXB и искать что-то другое? Ну, я могу написать метод "XMLNode toXML()" для всех классов, которые я не хочу (де)маршалировать, но...