Как добавить атрибуты xml в jaxb аннотированный класс XmlElementWrapper?
У меня есть класс с аннотацией XmlElementWrapper, например:
...
@XmlElementWrapper(name="myList")
@XmlElements({
@XmlElement(name="myElement") }
)
private List<SomeType> someList = new LinkedList();
...
Этот код создает XML как
<myList>
<myElement> </myElement>
<myElement> </myElement>
<myElement> </myElement>
</myList>
пока что так хорошо.
Но теперь мне нужно добавить атрибуты в тег списка, чтобы получить XML как
<myList number="2">
<myElement> </myElement>
<myElement> </myElement>
<myElement> </myElement>
</myList>
Есть ли "умный способ достичь этого без создания нового класса, который содержит список?
Ответы
Ответ 1
Я получил лучшее решение для вашего вопроса.
Для создания объекта Xml Java используйте следующий код:
import java.util.*;
import javax.xml.bind.annotation.*;
@XmlRootElement(name="myList")
public class Root {
private String number;
private List<String> someList;
@XmlAttribute(name="number")
public String getNumber() {
return number;
}
public void setNumber(String number) {
this.number = number;
}
@XmlElement(name="myElement")
public List<String> getSomeList() {
return someList;
}
public void setSomeList(List<String> someList) {
this.someList = someList;
}
public Root(String numValue,List<String> someListValue) {
this();
this.number = numValue;
this.someList = someListValue;
}
/**
*
*/
public Root() {
// TODO Auto-generated constructor stub
}
}
Чтобы запустить вышеуказанный код с помощью JAXB, используйте следующее:
import java.util.ArrayList;
import java.util.List;
import javax.xml.bind.*;
public class Demo {
public static void main(String[] args) throws Exception {
List<String> arg = new ArrayList<String>();
arg.add("FOO");
arg.add("BAR");
Root root = new Root("123", arg);
JAXBContext jc = JAXBContext.newInstance(Root.class);
Marshaller marshaller = jc.createMarshaller();
marshaller.marshal(root, System.out);
}
}
В результате вы получите следующий XML:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<myList number="123">
<myElement>FOO</myElement>
<myElement>BAR</myElement>
</myList>
Я думаю, что это более полезно вам.
Спасибо..
Ответ 2
MOXy Реализация JAXB (я ведущий технологии) имеет расширение (@XmlPath) для обработки этого случая:
import java.util.*;
import javax.xml.bind.annotation.*;
import org.eclipse.persistence.oxm.annotations.XmlPath;
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Root {
@XmlPath("myList/@number")
private int number;
@XmlElementWrapper(name="myList")
@XmlElement(name="myElement")
private List<String> someList = new LinkedList<String>();
public int getNumber() {
return number;
}
public void setNumber(int number) {
this.number = number;
}
public List<String> getSomeList() {
return someList;
}
public void setSomeList(List<String> someList) {
this.someList = someList;
}
}
Выведет следующий XML:
<?xml version="1.0" encoding="UTF-8"?>
<root>
<myList number="123">
<myElement>FOO</myElement>
<myElement>BAR</myElement>
</myList>
</root>
Когда этот код запущен:
import javax.xml.bind.*;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(Root.class);
Root root = new Root();
root.setNumber(123);
root.getSomeList().add("FOO");
root.getSomeList().add("BAR");
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(root, System.out);
}
}
Чтобы заставить это работать с использованием строго стандартного кода JAXB, вам нужно использовать XML-адаптер:
Примечание:
Чтобы использовать MOXy JAXB, вам нужно добавить файл с именем jaxb.properties с вашими классами моделей со следующей записью:
javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory
Ответ 3
Если вы не используете MOXy или просто хотите придерживаться стандартных аннотаций JAXB, вы можете продлить на Noby ответ, чтобы добавить поддержку для общего класса оболочки. Нискованный ответ только в настоящее время поддерживает список строк, но скажем, например, что вы собираетесь использовать один и тот же общий класс-оболочку для нескольких разных классов. В моем примере я хочу создать общий класс "PagedList", который будет маршировать к тому, что выглядит как список, но также содержит информацию о смещении страницы и об общем количестве элементов в неуправляемом списке.
Единственным недостатком этого решения является то, что вам нужно добавить дополнительные сопоставления @XmlElement для каждого типа класса, который будет обернут. В целом, хотя это, вероятно, лучшее решение, чем создание нового класса для каждого pagable элементов.
@XmlType
public class PagedList<T> {
@XmlAttribute
public int offset;
@XmlAttribute
public long total;
@XmlElements({
@XmlElement(name="order", type=Order.class),
@XmlElement(name="address", type=Address.class)
// additional as needed
})
public List<T> items;
}
@XmlRootElement(name="customer-profile")
public class CustomerProfile {
@XmlElement
public PagedList<Order> orders;
@XmlElement
public PagedList<Address> addresses;
}
Упорядочить этот пример поможет вам:
<customer-profile>
<order offset="1" total="100">
<order> ... </order>
<order> ... </order>
<order> ... </order>
...
</orders>
<addresses offset="1" total="5">
<address> ... </address>
<address> ... </address>
<address> ... </address>
<address> ... </address>
<address> ... </address>
<addresses>
</customer-profile>
Надеюсь, что это поможет. Это решение, на котором я остановился, по крайней мере.