Как комментировать список с помощью @XmlElement?
У меня есть следующая аннотация с помощью javax.xml.bind.annotation.XmlElement
:
@XmlElement
public List<String> getKeywords() {
return keywords;
}
Что создает следующий XML, когда я собираю некоторые примеры контента:
<keywords>keyword1</keywords>
<keywords>keyword2</keywords>
Я хотел бы получить следующий XML:
<keywords>
<keyword>keyword1</keyword>
<keyword>keyword2</keyword>
</keywords>
Какую аннотацию следует использовать?
Я пробовал
@XmlElementWrapper
@XmlElement(name="keyword")
Но тогда весь контент исчезает, и результат:
<keywords/>
То же самое происходит и при попытке переименования элемента:
@XmlElement(name="keyword")
Что я делаю неправильно?
UPDATE:
Вот обновленный полный код для класса в соответствии с первым ответом, но он все еще не работает (результатом является пустой список <keywords/>
при сопоставлении с XML):
import java.util.List;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlElementWrapper;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class Content {
private List<String> keywords;
public Content() {}
@XmlElementWrapper(name="keywords")
@XmlElement(name="keyword")
public List<String> getKeywords() {
return keywords;
}
public void setKeywords(List<String> keywords) {
this.keywords = keywords;
}
}
Я также попробовал следующее с тем же результатом:
import java.util.List;
import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlElementWrapper;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Content {
@XmlElementWrapper(name="keywords")
@XmlElement(name="keyword")
private List<String> keywords;
public Content() {}
public List<String> getKeywords() {
return keywords;
}
public void setKeywords(List<String> keywords) {
this.keywords = keywords;
}
}
Однако ключевые слова не пустые, поскольку следующее создает <keywords>keyword1</keywords><keywords>keyword2</keywords>
вместо пустого списка:
import java.util.List;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlElementWrapper;
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class Content {
private List<String> keywords;
public Content() {}
@XmlElement
public List<String> getKeywords() {
return keywords;
}
public void setKeywords(List<String> keywords) {
this.keywords = keywords;
}
}
Код для сортировки - это (JAX-RS):
import java.io.StringWriter;
import javax.ws.rs.Consumes;
import javax.ws.rs.Path;
import javax.ws.rs.POST;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
@Path("process")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_XML)
public class ContentHandler {
@POST
public Response process(Content content) {
StringWriter stringWriter = new StringWriter();
try {
JAXBContext jaxbContext = JAXBContext.newInstance(Content.class);
Marshaller jaxbMarshaller = jaxbContext.createMarshaller();
jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
jaxbMarshaller.marshal(content, stringWriter);
} catch (JAXBException e) {
return Response.serverError().entity(e.getMessage()).build();
}
return Response.ok(stringWriter.toString(), MediaType.APPLICATION_XML).build();
}
}
Ответы
Ответ 1
Вам нужно использовать @XmlElementWrapper
и @XmlElement
.
Модель Java
Содержание
import java.util.List;
import javax.xml.bind.annotation.*;
@XmlRootElement
public class Content {
private List<String> keywords;
public Content() {}
@XmlElementWrapper
@XmlElement(name="keyword")
public List<String> getKeywords() {
return keywords;
}
public void setKeywords(List<String> keywords) {
this.keywords = keywords;
}
}
Демо-код
Demo
import java.util.*;
import javax.xml.bind.*;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(Content.class);
List<String> strings = new ArrayList<String>(2);
strings.add("foo");
strings.add("bar");
Content content = new Content();
content.setKeywords(strings);
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(content, System.out);
}
}
Выход
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<content>
<keywords>
<keyword>foo</keyword>
<keyword>bar</keyword>
</keywords>
</content>
Для получения дополнительной информации
Ниже приведены ссылки на пару статей из моего блога, которые содержат дополнительную информацию:
Ответ 2
Используйте эту форму:
@XmlElementWrapper(name="keywords")
@XmlElement(name="keyword")
Обратите внимание, что если keywords
пусто, вы получите <keywords />
.
Иногда вам нужно добавить @XmlRootElement
в свой класс (зависит от контекста) и аннотации @XmlAccessorType(XmlAccessType.?)
. Обычно я использую @XmlAccessorType(XmlAccessType.FIELD)
и аннотирую свои поля @XmlElement
.
Ответ 3
Над ответом - Блейз Дохан полностью прав.
Еще один простой способ - даже если вы не пишете - @XmlElementWrapper
private List<String> keywords;
@XmlElementWrapper
@XmlElement(name="keyword")
public List<String> getKeywords() {
return keywords;
}
Вы можете использовать его таким образом - напишите XmlAccessorType на уровне класса, тогда имя XML-элемента будет таким же, как имя члена класса - ключевые слова
import java.util.List;
import javax.xml.bind.annotation.*;
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Content {
private List<String> keywords;
public Content() {}
public List<String> getKeywords() {
return keywords;
}
public void setKeywords(List<String> keywords) {
this.keywords = keywords;
}
}