使用泛型和基类时,Jaxb XML命名空间不正确

时间:2015-12-16 14:49:20

标签: java jaxb jaxb2

我在两个名称空间中有四个类,两个在一个包中用作另一个包中的两个类的基类,例如:

@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>

1 个答案:

答案 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的人可能无法识别这些元素名称。