Удалите префикс пространства имен во время сортировки JAXB
У меня есть объекты JAXB, созданные из схемы. При сортировке элементы xml аннотируются с помощью ns2. Я перепробовал все варианты, которые существуют в сети для этой проблемы, но ни один из них не работает. Я не могу изменить мою схему или изменить package-info.java. Пожалуйста помоги
6 ответов
После долгих исследований и переделок мне наконец удалось найти решение этой проблемы. Пожалуйста, примите мои извинения за то, что я не публикую ссылки на оригинальные ссылки - их много, и я не делал заметок - но этот, безусловно, был полезен.
Мое решение использует фильтрацию XMLStreamWriter
который применяет пустой контекст пространства имен.
public class NoNamesWriter extends DelegatingXMLStreamWriter {
private static final NamespaceContext emptyNamespaceContext = new NamespaceContext() {
@Override
public String getNamespaceURI(String prefix) {
return "";
}
@Override
public String getPrefix(String namespaceURI) {
return "";
}
@Override
public Iterator getPrefixes(String namespaceURI) {
return null;
}
};
public static XMLStreamWriter filter(Writer writer) throws XMLStreamException {
return new NoNamesWriter(XMLOutputFactory.newInstance().createXMLStreamWriter(writer));
}
public NoNamesWriter(XMLStreamWriter writer) {
super(writer);
}
@Override
public NamespaceContext getNamespaceContext() {
return emptyNamespaceContext;
}
}
Вы можете найти DelegatingXMLStreamWriter
здесь
Затем вы можете отфильтровать маршаллинг xml с помощью:
// Filter the output to remove namespaces.
m.marshal(it, NoNamesWriter.filter(writer));
Я уверен, что есть более эффективные механизмы, но я знаю, что этот работает.
Для меня только изменение класса package-info.java сработало как талисман, именно так, как сказал zatziky:
package-info.java
@javax.xml.bind.annotation.XmlSchema
(namespace = "http://example.com",
xmlns = {@XmlNs(prefix = "", namespaceURI = "http://example.com")},
elementFormDefault = javax.xml.bind.annotation.XmlNsForm.QUALIFIED)
package my.package;
import javax.xml.bind.annotation.XmlNs;
Вы можете позволить пространствам имен записываться только один раз. Вам понадобится прокси-класс XMLStreamWriter и package-info.java. Тогда вы будете делать в своем коде:
StringWriter stringWriter = new StringWriter();
XMLStreamWriter writer = new Wrapper((XMLStreamWriter) XMLOutputFactory
.newInstance().createXMLStreamWriter(stringWriter));
JAXBContext jaxbContext = JAXBContext.newInstance(Collection.class);
Marshaller jaxbMarshaller = jaxbContext.createMarshaller();
jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE);
jaxbMarshaller.marshal(books, writer);
System.out.println(stringWriter.toString());
Прокси-класс (важный метод - writeNamespace):
class WrapperXMLStreamWriter implements XMLStreamWriter {
private final XMLStreamWriter writer;
public WrapperXMLStreamWriter(XMLStreamWriter writer) {
this.writer = writer;
}
//keeps track of what namespaces were used so that not to
//write them more than once
private List<String> namespaces = new ArrayList<String>();
public void init(){
namespaces.clear();
}
public void writeStartElement(String localName) throws XMLStreamException {
init();
writer.writeStartElement(localName);
}
public void writeStartElement(String namespaceURI, String localName) throws XMLStreamException {
init();
writer.writeStartElement(namespaceURI, localName);
}
public void writeStartElement(String prefix, String localName, String namespaceURI) throws XMLStreamException {
init();
writer.writeStartElement(prefix, localName, namespaceURI);
}
public void writeNamespace(String prefix, String namespaceURI) throws XMLStreamException {
if(namespaces.contains(namespaceURI)){
return;
}
namespaces.add(namespaceURI);
writer.writeNamespace(prefix, namespaceURI);
}
// .. other delegation method, always the same pattern: writer.method() ...
}
package-info.java:
@XmlSchema(elementFormDefault=XmlNsForm.QUALIFIED, attributeFormDefault=XmlNsForm.UNQUALIFIED ,
xmlns = {
@XmlNs(namespaceURI = "http://www.w3.org/2001/XMLSchema-instance", prefix = "xsi")})
package your.package;
import javax.xml.bind.annotation.XmlNs;
import javax.xml.bind.annotation.XmlNsForm;
import javax.xml.bind.annotation.XmlSchema;
Вы можете использовать NamespacePrefixMapper
расширение для управления префиксами пространства имен для вашего варианта использования. Одно и то же расширение поддерживается как эталонной реализацией JAXB, так и EclipseLink JAXB (MOXy).
Каждое решение требует сложной перезаписи или аннотаций, которые, похоже, не работают с последней версией. Я использую более простой подход, просто заменяя раздражающие пространства имен. Я бы хотел, чтобы Google & Co использовала JSON и избавилась от XML.
kml.marshal(file);
String kmlContent = FileUtils.readFileToString(file, "UTF-8");
kmlContent = kmlContent.replaceAll("ns2:","").replace("<kml xmlns:ns2=\"http://www.opengis.net/kml/2.2\" xmlns:ns3=\"http://www.w3.org/2005/Atom\" xmlns:ns4=\"urn:oasis:names:tc:ciq:xsdschema:xAL:2.0\" xmlns:ns5=\"http://www.google.com/kml/ext/2.2\">", "<kml>");
FileUtils.write(file, kmlContent, "UTF-8");
Я получил рабочий результат, установив свойство:
marshaller.setProperty(Marshaller.JAXB_SCHEMA_LOCATION, "");