Настройка коллекции полиморфных объектов для работы в JAXB2

Я переключаюсь с Castor на JAXB2 для выполнения маршалинга/демаршалинга между объектами XML и Java. У меня проблема с настройкой коллекции полиморфных объектов.

Пример XML

<project name="test project">
    <orange name="fruit orange" orangeKey="100" />
    <apple name="fruit apple" appleKey="200" />
    <orange name="fruit orange again" orangeKey="500" />
</project>

Класс проекта

Список oranges работает нормально, я вижу в списке 2 апельсина. Но я не уверен, как настроить fruitList. В fruitList должно быть 3 фрукта: 2 апельсина и 1 яблоко.

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Project {

    @XmlAttribute
    private String          name;

    @XmlElement(name = "orange")
    private List<Orange>    oranges     = new ArrayList<Orange>();

    // Not sure how to configure this... help!
    private List<Fruit>     fruitList   = new ArrayList<Fruit>();
}

Класс фруктов

Fruit — это абстрактный класс. По какой-то причине определение этого класса как абстрактного вызывает много проблем.

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public abstract class Fruit {

    @XmlAttribute
    private String  name;
}

Оранжевый класс

public class Orange extends Fruit {

    @XmlAttribute
    private String  orangeKey;
}

Класс Apple

public class Apple extends Fruit {

    @XmlAttribute
    private String  appleKey;
}

Как настроить fruitList в Project, чтобы добиться того, чего я хочу здесь?

Спасибо большое!


person limc    schedule 03.02.2011    source источник


Ответы (3)


Вы хотите использовать @XmlElementRef, это соответствует концепции групп замещения схемы XML, которая соответствует вашему вопросу.

Шаг 1. Использование @XmlElementRef

Свойство fruitList аннотировано @XmlElementRef:

import java.util.ArrayList;
import java.util.List;

import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlElementRef;
import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Project {

    @XmlAttribute
    private String name;

    @XmlElementRef
    private List<Fruit> fruitList = new ArrayList<Fruit>();

}

Шаг 2. Аннотируйте Apple и Orange с помощью @XmlRootElement

import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Apple extends Fruit {

    @XmlAttribute
    private String  appleKey;

}

а также

import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Orange extends Fruit {

    @XmlAttribute
    private String  orangeKey;

}

Демонстрационный код

Для демонстрации решения можно использовать следующий код:

import java.io.File;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;

public class Demo {

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

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        Project project = (Project) unmarshaller.unmarshal(new File("input.xml"));

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

}

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

person bdoughan    schedule 03.02.2011
comment
Я пробую ваш подход... Я аннотировал свойство fruitList с помощью @XmlElementRef и аннотировал классы Orange и Apple с помощью @XmlRootElement. Я запустил код, и fruitList оказалось пустым. Что я здесь делаю неправильно? Спасибо. - person limc; 03.02.2011
comment
Я понял проблему, я должен добавить Orange.class и Apple.class в JAXBContext.newInstance(..) вместе с Project. Работает сейчас. Спасибо. - person limc; 03.02.2011

После возни... Думаю, теперь у меня все работает: -

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Project {

    @XmlAttribute
    private String      name;

    // this has to be commented out, or else oranges won't show up in fruitList
    // @XmlElement(name = "orange")
    // private List<Orange> oranges = new ArrayList<Orange>();

    @XmlElements({
            @XmlElement(name = "orange", type = Orange.class),
            @XmlElement(name = "apple", type = Apple.class)
    })
    private List<Fruit> fruitList   = new ArrayList<Fruit>();

}
person limc    schedule 03.02.2011
comment
@XmlElements будет работать, но каждый раз, когда вы добавляете новый подкласс, вам придется редактировать класс Project. Если вместо этого вы используете @XmlElementRef, вам просто нужно будет аннотировать новый подкласс с помощью @XmlRootElement. @XmlElements больше подходит для представления структур выбора: bdoughan.blogspot .com/2010/10/ - person bdoughan; 03.02.2011

Поместите аннотацию XmlAnyElement:

@XmlAnyElement(lax = true)
private List<Fruit>     fruitList   = new ArrayList<Fruit>();
person Tarlog    schedule 03.02.2011
comment
Это не сработало... fruitList содержит один apple типа ElementNSImpl. - person limc; 03.02.2011
comment
Для @XmlAnyElement вам нужно будет добавить @XmlRootElement к каждому из подклассов. Однако это больше относится к разделам с подстановочными знаками. Поскольку дочерние объекты находятся в отношениях наследования, лучше использовать @XmlElementRef. Дополнительные сведения о @XmlAnyElement см. на странице bdoughan.blogspot. ком/2010/08/ - person bdoughan; 03.02.2011