是否可以以编程方式配置 JAXB?

2022-09-02 13:36:16

假设我有两个 JavaBean 和 .PersonAddress

如果我创建 Person 对象的列表,我想封送如下:

<persons>
  <person>...</person>
</persons>

可以使用此处描述的技术:使用 JAXB 取消元帅/封送 List<String>

通过使用 和 对 JaxbList 进行注释,则可以封送至上面的 XML。@XmlRootElement(name = "persons")@XmlElement(name = "person")

但是,如果能够重用同一类来封送对象列表,那就太好了。实际上,我会有很多其他类型的豆子。我可以像这样:JaxbList<T>Address

<list>
   <item xsi:type="person" xmlns:xsi="http://www.w2.org/2001/XmlSchema-instance"></item>
</list>

但是,理想情况下,最好将“list”替换为类名的复数版本,将“item”替换为类名。

那么,是否可以在运行时以编程方式配置JaxbContext或其他东西,并基本上设置内部和的值?name@XmlRootElement@XmlElement

或者任何其他方法来让它工作,而不必为每个bean类型编写单独的实现?也许XmlJavaTypeAdapter可以实现这种事情?JaxbList

更新@Blaise下面接受的Doughan的解决方案效果很好。对于我的用例,我需要直接从Java对象到XML,这是有效的(请注意,这不是我的完整实现,它只是用于演示的伪代码):

    //JAXBContext is thread safe and so create it in constructor or 
    //setter or wherever:
    ... 
    JAXBContext jc = JAXBContext.newInstance(Wrapper.class, clazz);
    ... 

    public String marshal(List<T> things, Class clazz) {

      //configure JAXB and marshaller     
      Marshaller m = jc.createMarshaller();
      m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

      //Create wrapper based on generic list of objects
      Wrapper<T> wrapper = new Wrapper<T>(things);
      JAXBElement<Wrapper> wrapperJAXBElement = new JAXBElement<Wrapper>(new QName(clazz.getSimpleName().toLowerCase()+"s"), Wrapper.class, wrapper);

      StringWriter result = new StringWriter();
      //marshal!
      m.marshal(wrapperJAXBElement, result);

      return result.toString();

    }

答案 1

您可以创建如下所示的通用对象:Wrapper

包装纸

可以创建一个泛型包装类,其属性用 注释为 。用于填充此列表的对象的类型将基于其根元素(请参见:http://blog.bdoughan.com/2010/08/using-xmlanyelement-to-build-generic.html)。List@XmlAnyElement(lax=true)

package forum13272288;

import java.util.*;
import javax.xml.bind.annotation.XmlAnyElement;

public class Wrapper<T> {

    private List<T> items = new ArrayList<T>();

    @XmlAnyElement(lax=true)
    public List<T> getItems() {
        return items;
    }

}

地址

您需要使用 注释列表的可能内容。@XmlRootElement

package forum13272288;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Address {

}

package forum13272288;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Person {

}

演示

下面的演示代码演示如何使用该类。由于根元素可能不同,因此您需要指定要取消绑定到包装器类。或者,您可以利用注释将多个根元素与包装类相关联(请参见:http://blog.bdoughan.com/2012/07/jaxb-and-root-elements.html)。Wrapper@XmlElementDecl

package forum13272288;

import javax.xml.bind.*;
import javax.xml.transform.stream.StreamSource;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(Wrapper.class, Person.class, Address.class);
        Unmarshaller unmarshaller = jc.createUnmarshaller();
        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

        StreamSource personsXML = new StreamSource("src/forum13272288/persons.xml");
        JAXBElement<Wrapper> wrapper1 = unmarshaller.unmarshal(personsXML, Wrapper.class);
        marshaller.marshal(wrapper1, System.out);

        StreamSource addressesXML = new StreamSource("src/forum13272288/addresses.xml");
        JAXBElement<Wrapper> wrapper2 = unmarshaller.unmarshal(addressesXML, Wrapper.class);
        marshaller.marshal(wrapper2, System.out);
    }

}

输出

以下是运行演示代码的输出。文件和看起来就像有相应的输出。persons.xmladdresses.xml

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<persons>
    <person/>
    <person/>
</persons>
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<addresses>
    <address/>
    <address/>
</addresses>

详细信息


答案 2

推荐