7

似乎 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 或从 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 并寻找其他东西?我,好吧,可以为我不想(de)编组的所有类编写“XMLNode toXML()”方法,但是......

4

1 回答 1

10

试试下面的@XmlAnyElement(lax=true)。该lax标志告诉 JAXB (JSR-222) 实现根据元素@XmlRootElement@XmlElementDecl注释将元素与域对象匹配。没有它,内容将被视为 DOM 节点。

@XmlRootElement
public class Holder {
    private List<IFoo> things;

    public Holder() {
        things = new ArrayList<>();
    }

    @XmlElementWrapper
    @XmlAnyElement(lax=true)
    public List<IFoo> getThings() {
        return things;
    }

    public void addThing(IFoo thing) {
        things.add(thing);
    }
}

了解更多信息

于 2012-06-20T20:53:15.083 回答