假设我有两个JavaBeans Person
和Address
。
如果我创建一个Person对象列表,我想编组这样的东西:
<persons>
<person>...</person>
</persons>
可以使用此处描述的技术: Using JAXB to unmarshal/marshal a List<String>
通过使用@XmlRootElement(name = "persons")
和@XmlElement(name = "person")
注释JaxbList,可以编组上面的XML。
但是,能够重用相同的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类型编写JaxbList
的单独实现?也许XmlJavaTypeAdapter可以实现这种功能吗?
更新 @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();
}
答案 0 :(得分:17)
您可以创建一个通用的Wrapper
对象,如下所示:
<强>包装强>
您可以创建一个通用的包装类,其List
属性使用@XmlAnyElement(lax=true)
注释。用于填充此列表的对象类型将基于其根元素(请参阅:http://blog.bdoughan.com/2010/08/using-xmlanyelement-to-build-generic.html)。
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
注释将多个根元素与包装类相关联(请参阅:http://blog.bdoughan.com/2012/07/jaxb-and-root-elements.html)。
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
看起来就像是相应的输出。
<?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>
了解更多信息