JAXB-Eclipselink: XmlRootElement и наследование - PullRequest
2 голосов
/ 13 января 2012

Используя Eclipselink / MOXy 2.3, у меня есть следующий вариант использования при маршалинге в XML:

abstract class MyAbstract {
}

class MyImpl extends MyAbstract {
}

class A {

    private MyAbstract myAbstract;

    // MyImpl is behind this
    public MyAbstract getMyAbstract() {
        return myAbstract;
    }

}

У меня есть следующее отображение, определенное в oxm.xml:

<java-type name="foo.MyAbstract" xml-accessor-type="NONE">
    <xml-see-also>
        foo.MyImpl
    </xml-see-also>
</java-type>

<java-type name="foo.MyImpl">
    <xml-root-element name="MyImpl" />
</java-type>

<java-type name="bar.A" xml-accessor-type="NONE">
    <xml-root-element name="A" />
    <java-attributes>
        <xml-element java-attribute="myAbstract" type="foo.MyAbstract" />
    </java-attributes>
</java-type>

Теперь это приводит к:

<A>
    <myAbstract xsi:type="myImpl">
        <!-- Mapped members of MyImpl + MyAbstract -->
    </myAbstract>
</A>

Поскольку я не хотел, чтобы имя свойства в экспортированном XML-файле было изменено:

<java-type name="bar.A" xml-accessor-type="NONE">
    <xml-root-element name="A" />
    <java-attributes>
        <xml-element java-attribute="myAbstract" type="foo.MyAbstract" xml-path="."/>
    </java-attributes>
</java-type>

что привело к:

<A>
    <!-- Members of MyImpl + MyAbstract marshalled without any wrapping element-->
</A>

Что я хочу, это:

 <A>
    <MyImpl>
        <!-- Members of MyImpl + MyAbstract -->
    </MyImpl>
 </A>

Вопрос: как мне этого добиться? MOXy просто игнорирует мой XmlRootElement на MyImpl ...

EDIT:

Попытка, предложенная Блейзом, дает мне следующее исключение:

Exception [EclipseLink-60] (Eclipse Persistence Services - 2.3.2.v20111125-r10461):   
org.eclipse.persistence.exceptions.DescriptorException
The method [] or [getMyAbstract] is not defined in the object [bar.A].

Теперь для этого нужна дополнительная информация, которую я пропустил раньше, потому что я думал, что это не имеет отношения

Класс A - это интерфейс, который определяет: public X getMyAbstract(); MyAbstract реализует X (именно поэтому я добавил атрибут type в отображение для интерфейса A).

Таким образом, использование xml-element-ref MOXy больше не «видит» геттер, а xml-element делает.

1 Ответ

2 голосов
/ 14 января 2012

Сопоставление, которое вы ищете, @XmlElementRef.Это соответствует концепции групп замещения в XML-схеме.

bar / oxm.xml

Ниже приведен документ внешнего сопоставления для пакета bar.Обратите внимание, как свойство myAbstract сопоставляется с xml-element-ref, которое представляет собой XML-представление @XmlElementRef

<?xml version="1.0" encoding="UTF-8"?>
<xml-bindings
    xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
    package-name="bar">
    <java-types>
        <java-type name="A" xml-accessor-type="NONE">
            <xml-root-element name="A" />
            <java-attributes>
                <xml-element-ref java-attribute="myAbstract"/>
            </java-attributes>
        </java-type>
    </java-types>
</xml-bindings>

foo / oxm.xml

Ниже приведен внешнийФайл метаданных для пакета foo:

<?xml version="1.0" encoding="UTF-8"?>
<xml-bindings
    xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
    package-name="foo">
    <java-types>
        <java-type name="MyAbstract" xml-accessor-type="NONE">
            <xml-see-also>
                foo.MyImpl
            </xml-see-also>
        </java-type>
        <java-type name="MyImpl">
            <xml-root-element name="MyImpl" />
        </java-type>
    </java-types>
</xml-bindings>

Demo

Ниже приведен демонстрационный код для этого примера:

package forum8853855;

import java.util.*;
import javax.xml.bind.*;    
import org.eclipse.persistence.jaxb.JAXBContextFactory;
import bar.A;
import foo.MyImpl;

public class Demo {

    public static void main(String[] args) throws Exception {
        List<String> oxm = new ArrayList<String>(2);
        oxm.add("foo/oxm.xml");
        oxm.add("bar/oxm.xml");

        Map<String, Object> properties = new HashMap<String, Object>(1);
        properties.put(JAXBContextFactory.ECLIPSELINK_OXM_XML_KEY, oxm);

        JAXBContext jc = JAXBContext.newInstance(new Class[] {A.class}, properties);
        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

        A a = new A();
        a.setMyAbstract(new MyImpl());
        marshaller.marshal(a, System.out);
    }

}

Вывод

<?xml version="1.0" encoding="UTF-8"?>
<A>
   <MyImpl/>
</A>

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

...