Удалите префикс пространства имен во время сортировки 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, "");
Другие вопросы по тегам