说我有两个JavaBeans Person和Address。
Person
Address
如果创建一个Person对象的列表,我想编组成这样的东西:
<persons> <person>...</person> </persons>
可以使用这里描述的技术: 使用JAXB解组/编组List
通过使用@XmlRootElement(name = "persons")和注释JaxbList,@XmlElement(name ="person")可以将其编组为上述XML。
@XmlRootElement(name = "persons")
@XmlElement(name ="person")
但是,能够重用相同的JaxbList<T>类来封送Address对象列表也很好。实际上,我将有许多其他类型的bean。我可以这样:
JaxbList<T>
<list> <item xsi:type="person" xmlns:xsi="http://www.w2.org/2001/XmlSchema-instance"></item> </list>
但是,理想情况下,最好用类名的复数形式替换“列表”,用类名替换“ item”。
因此,是否有可能在运行时以编程方式配置JaxbContext或其他内容并本质上设置nameinside @XmlRootElement和的值@XmlElement?
name
@XmlRootElement
@XmlElement
还是通过其他方法无需编写JaxbList每种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(); }
您可以创建如下的通用Wrapper对象:
Wrapper
包装纸
您可以创建一个通用包装器类,其List属性带有注释@XmlAnyElement(lax=true)。用于填充此列表的对象的类型将基于其根元素
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 { }
演示版
下面的演示代码演示了如何使用Wrapper该类。由于根元素可以不同,因此您需要指定要解组包装类。或者,您可以利用@XmlElementDecl注释将多个根元素与包装类相关联
@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.xml和addresses.xml看起来就像那里相应的输出。
persons.xml
addresses.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>