gpt4 book ai didi

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

转载 作者:太空狗 更新时间:2023-10-29 22:34:42 25 4
gpt4 key购买 nike

假设我有两个 JavaBeans PersonAddress .

如果我创建一个 Person 对象列表,我想编码成这样的东西:

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

可以使用此处描述的技术: Using JAXB to unmarshal/marshal a List<String>

通过使用 @XmlRootElement(name = "persons") 注释 JaxbList和 @XmlElement(name = "person") ,那么就可以编码到上面的 XML。

但是,能够重复使用相同的 JaxbList<T> 会很好类还编码 Address 的列表对象。实际上,我会有很多其他类型的 bean 。我可以选择类似的东西:

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

但是,理想情况下,最好用类名的复数形式替换“list”,用类名替换“item”。

那么,是否可以在运行时以编程方式配置 JaxbContext 或其他东西,并实质上设置 name 的值?里面@XmlRootElement@XmlElement

或任何其他无需编写 JaxbList 的单独实现即可使其正常工作的方法对于每种 bean 类?也许 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();

}

最佳答案

您可以创建一个通用的 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.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>

了解更多信息

关于java - 是否可以以编程方式配置 JAXB?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13272288/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com