Можно ли программно настроить JAXB?

Скажем, у меня два JavaBeans Person и Address.

Если я создаю список объектов Person, я хотел бы маршалировать что-то вроде этого:

<persons>
  <person>...</person>
</persons>

Можно использовать описанную здесь методику: Использование JAXB для отмены/маршалирования списка <String>

Аннотируя JaxbList с помощью @XmlRootElement(name = "persons") и @XmlElement(name = "person"), тогда можно перенести на XML выше.

Но было бы неплохо повторить использование одного и того же класса JaxbList<T>, чтобы также объединить список объектов Address. И на самом деле у меня будет много других типов beans. Я могу пойти с чем-то вроде:

<list>
   <item xsi:type="person" xmlns:xsi="http://www.w2.org/2001/XmlSchema-instance"></item>
</list>

Но, в идеале, было бы неплохо, если бы он заменил "список" на множественную версию имени класса и "item" на имя класса.

Итак, возможно ли программно настроить JaxbContext или что-то во время выполнения и по существу установить значение name внутри @XmlRootElement и @XmlElement?

Или любой другой способ получить эту работу без необходимости писать отдельную реализацию JaxbList для каждого типа bean? Возможно, XmlJavaTypeAdapter может достичь такого рода вещей?

Обновление Решение @Blaise Doughan, принятое ниже, отлично работает. Для моего варианта использования мне нужно было перейти прямо из объекта Java к XML, вот что сработало (обратите внимание, что это не моя полная реализация, это просто псевдо-код для демонстрации):

    //JAXBContext is thread safe and so create it in constructor or 
    //setter or wherever:
    ... 
    JAXBContext jc = JAXBContext.newInstance(Wrapper.class, clazz);
    ... 

    public String marshal(List<T> things, Class clazz) {

      //configure JAXB and marshaller     
      Marshaller m = jc.createMarshaller();
      m.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

      //Create wrapper based on generic list of objects
      Wrapper<T> wrapper = new Wrapper<T>(things);
      JAXBElement<Wrapper> wrapperJAXBElement = new JAXBElement<Wrapper>(new QName(clazz.getSimpleName().toLowerCase()+"s"), Wrapper.class, wrapper);

      StringWriter result = new StringWriter();
      //marshal!
      m.marshal(wrapperJAXBElement, result);

      return result.toString();

    }

Ответы

Ответ 1

Вы можете создать общий Wrapper объект, например:

Упаковочный

Вы можете создать общий класс-оболочку с свойством List, аннотированным с помощью @XmlAnyElement(lax=true). Тип объекта, который используется для заполнения этого списка, будет основываться на его корневом элементе (см. http://blog.bdoughan.com/2010/08/using-xmlanyelement-to-build-generic.html).

package forum13272288;

import java.util.*;
import javax.xml.bind.annotation.XmlAnyElement;

public class Wrapper<T> {

    private List<T> items = new ArrayList<T>();

    @XmlAnyElement(lax=true)
    public List<T> getItems() {
        return items;
    }

}

Адрес

Вам нужно будет аннотировать возможное содержимое списка с помощью @XmlRootElement.

package forum13272288;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Address {

}

Person

package forum13272288;

import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement
public class Person {

}

Demo

Демо-код ниже демонстрирует, как использовать класс Wrapper. Поскольку корневой элемент может быть другим, вам нужно будет указать, что вы хотите развязать класс оболочки. В качестве альтернативы вы можете использовать аннотацию @XmlElementDecl для связывания нескольких корневых элементов с классом-оболочкой (см. http://blog.bdoughan.com/2012/07/jaxb-and-root-elements.html).

package forum13272288;

import javax.xml.bind.*;
import javax.xml.transform.stream.StreamSource;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(Wrapper.class, Person.class, Address.class);
        Unmarshaller unmarshaller = jc.createUnmarshaller();
        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);

        StreamSource personsXML = new StreamSource("src/forum13272288/persons.xml");
        JAXBElement<Wrapper> wrapper1 = unmarshaller.unmarshal(personsXML, Wrapper.class);
        marshaller.marshal(wrapper1, System.out);

        StreamSource addressesXML = new StreamSource("src/forum13272288/addresses.xml");
        JAXBElement<Wrapper> wrapper2 = unmarshaller.unmarshal(addressesXML, Wrapper.class);
        marshaller.marshal(wrapper2, System.out);
    }

}

Выход

Ниже приведен результат запуска демонстрационного кода. Файлы persons.xml и addresses.xml выглядят так же, как и соответствующий вывод.

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<persons>
    <person/>
    <person/>
</persons>
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<addresses>
    <address/>
    <address/>
</addresses>

Дополнительная информация