Вставьте пользовательский сеттер в плагин maven-jaxb2 - PullRequest
3 голосов
/ 07 сентября 2011

Я использую org.jvnet.jaxb2.maven2: maven-jaxb2-plugin для создания POJO из файлов схемы XSD.

Теперь я хочу вставить что-то вроде пользовательскогосеттер.Он должен обрезать все строки и удалить определенные символы.

Знаете ли вы, как это сделать?


Файл XJB:

<?xml version="1.0" encoding="UTF-8"?>
<jaxb:bindings version="2.0" xmlns:jaxb="http://java.sun.com/xml/ns/jaxb" xmlns:xsd="http://www.w3.org/2001/XMLSchema"
    xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:xjc="http://java.sun.com/xml/ns/jaxb/xjc">
    <jaxb:bindings schemaLocation="my-schema-xml4.xsd" node="/xs:schema">
        <xjc:javaType name="java.lang.String" adapter="my.StringAdapter" />
    </jaxb:bindings>
</jaxb:bindings>

Решение длятипы привязки Java:

<?xml version="1.0" encoding="UTF-8" ?>
<bindings version="2.0" xmlns="http://java.sun.com/xml/ns/jaxb" xmlns:xs="http://www.w3.org/2001/XMLSchema"
    xmlns:xjc="http://java.sun.com/xml/ns/jaxb/xjc">
    <bindings schemaLocation="mySchema-xml4.xsd" node="/xs:schema">
        <globalBindings>
            <xjc:javaType name="java.lang.String" xmlType="xs:string"
                adapter="com.name.MyAdapter" />
                    <xjc:javaType name="java.lang.String" xmlType="xs:anySimpleType"
                adapter="com.name.MyAdapter" />
        </globalBindings>
    </bindings>
</bindings>

Но @XmlJavaTypeAdapter по-прежнему не добавляется к свойству content в узлах со смешанным содержимым, хотя свойство имеет тип java.lang.String.

Ответы [ 4 ]

1 голос
/ 07 сентября 2011

Я думаю, что лучший способ добиться этого - реализовать собственный XmlAdapter и настроить его с помощью настройки свойств. Вы можете сделать это с помощью стандартной настройки jaxb:property или с помощью плагина annotate .

jaxb:property

  <jaxb:property>
    <jaxb:baseType>
      <xjc:javaType name="java.lang.String"
        adapter="com.acme.foo.MyAdapter"/>
    </jaxb:baseType>
  </jaxb:property>

Плагин аннотирования:

  <annox:annotate target="field">
    <annox:annotate
      annox:class="javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter"
      value="com.acme.foo.MyAdapter"/>
  </annox:annotate>

Смотрите пример проекта здесь:

https://svn.java.net/svn/jaxb2-commons~svn/basics/trunk/tests/one/

0 голосов
/ 16 сентября 2011

По моему мнению, вы должны использовать AOP, например, Pick Spring AOP, методы Intercept Adapter, чтобы иметь логику обрезки / обрезки.Фактически, теперь это может быть общая логика, которая будет применяться ко всем типам строк.Если это звучит убедительно, я могу помочь с кодом

0 голосов
/ 12 сентября 2011

Альтернативным подходом будет использование StAX StreamReaderDelegate для манипулирования текстом XML до его получения реализацией JAXB.

Демо

Ваш код будет выглядеть примерно так: Вы бы реализовали StreamReaderDelegate для управления текстом, возвращаемым из текстовых событий:

package forum7329881;

import java.io.FileInputStream;
import javax.xml.bind.*;
import javax.xml.stream.*;
import javax.xml.stream.util.StreamReaderDelegate;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(Customer.class);

        XMLInputFactory xif = XMLInputFactory.newInstance();
        XMLStreamReader xsr = xif.createXMLStreamReader(new FileInputStream("src/forum7329881/input.xml"));
        xsr = new MyStreamReaderDelegate(xsr);

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        Customer customer = (Customer) unmarshaller.unmarshal(xsr);

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(customer, System.out);
    }

    private static class MyStreamReaderDelegate extends StreamReaderDelegate {

        public MyStreamReaderDelegate(XMLStreamReader xsr) {
            super(xsr);
        }


        @Override
        public String getText() {
            return super.getText().trim();
        }

        @Override
        public char[] getTextCharacters() {
            return getText().toCharArray();
        }

        @Override
        public int getTextLength() {
            return getText().length();
        }

        @Override
        public int getTextStart() {
            return 0;
        }

    }

}

Input.xml

Ниже приведен пример ввода, который содержит пробелы в текстовых узлах:

<customer Id="1">
    <name>    Jane Doe    </name>
    <address>
        <street>    123 A Street    </street>
    </address>
</customer>

выход

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<customer>
    <address>
        <street>123 A Street A</street>
    </address>
    <name>Jane Doe</name>
</customer>

Клиент

package forum7329881;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Customer {

    private String name;
    private Address address;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Address getAddress() {
        return address;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

}

Адрес

package forum7329881;

public class Address {

    private String street;

    public String getStreet() {
        return street;
    }

    public void setStreet(String street) {
        this.street = street;
    }

}

Для получения дополнительной информации

Ниже приведена ссылка на более подробный пример, где я использовал StAX StreamReaderDelegate для поддержки демаршаллинга без учета регистра:

0 голосов
/ 10 сентября 2011

Если ваш XSD не изменится в будущем, просто сгенерируйте POJO, закомментируйте плагин JAXB и измените источник вручную.

Если ваш XSD меняется редко , вы все равно можете использовать это решение с системой контроля версий. Зафиксируйте сгенерированный источник, измените POJO и зафиксируйте изменения. В следующий раз, когда XSD будет изменен, сгенерируйте патч между двумя коммитами, сгенерируйте POJO из нового XSD и примените патч. Я нашел это решение намного проще в тех случаях, когда XSD почти никогда не меняется.

...