JAXB @XmlAdapter: Карта -> Адаптер списка? (только маршал)

У меня есть класс Map<String, String>.
Первое, что приходит в голову всем, — это преобразовать его в класс List<Pair<String,String>> (где Pair является пользовательским классом).

Я пробовал @XmlAdapter вот так:

public class MapPropertiesAdapter extends XmlAdapter<List<Property>, Map<String,String>> { ... }

Но Eclipse MOXy, реализация JAXB, которую я использую, закончилась ошибкой ClassCastException — «невозможно преобразовать HashMap в Collection».

Поддерживается ли это преобразование JAXB? Или я пропустил какую-то часть документации, которая объясняет, почему это не так?

PS: я хотел получить XML следующим образом:

<properties>
    <property name="protocol"/>
    <property name="marshaller"/>
    <property name="unmarshaller"/>
    <property name="timeout"/>
    ...
</properties>

Я получил это, только должен был использовать промежуточный класс. Также описано в разделе Обработка NPE в XMLCompositeObjectMappingNodeValue.marshalSingleValue( XMLCompositeObjectMappingNodeValue.java:161)


person Ondra Žižka    schedule 10.06.2013    source источник


Ответы (1)


Вместо того, чтобы адаптировать Map к List, вы должны адаптировать его к объекту, который имеет свойство List.

XmlAdapter (MapPropertiesAdapter)

import java.util.*;
import java.util.Map.Entry;
import javax.xml.bind.annotation.*;
import javax.xml.bind.annotation.adapters.XmlAdapter;

public class MapPropertiesAdapter extends XmlAdapter<MapPropertiesAdapter.AdaptedProperties, Map<String, String>>{

    public static class AdaptedProperties {
        public List<Property> property = new ArrayList<Property>();
    }

    public static class Property {
        @XmlAttribute
        public String name;

        @XmlValue
        public String value;
    }

    @Override
    public Map<String, String> unmarshal(AdaptedProperties adaptedProperties) throws Exception {
        if(null == adaptedProperties) {
            return null;
        }
        Map<String, String> map = new HashMap<String, String>(adaptedProperties.property.size());
        for(Property property : adaptedProperties.property) {
            map.put(property.name, property.value);
        }
        return map;
    }

    @Override
    public AdaptedProperties marshal(Map<String, String> map) throws Exception {
        if(null == map) {
            return null;
        }
        AdaptedProperties adaptedProperties = new AdaptedProperties();
        for(Entry<String,String> entry : map.entrySet()) {
            Property property = new Property();
            property.name = entry.getKey();
            property.value = entry.getValue();
            adaptedProperties.property.add(property);
        }
        return adaptedProperties;
    }

}

Модель домена (корневой)

Ниже представлен объект модели со свойством Map. Аннотация @XmlJavaTypeAdapter используется для указания файла XmlAdapter.

import java.util.Map;
import javax.xml.bind.annotation.*;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;

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

    @XmlJavaTypeAdapter(MapPropertiesAdapter.class)
    private Map<String, String> properties;

}

Демо

import java.io.File;
import javax.xml.bind.*;

public class Demo {

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

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        File xml = new File("src/forum17024050/input.xml");
        Root root = (Root) unmarshaller.unmarshal(xml);

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

}

входной.xml/выходной

<?xml version="1.0" encoding="UTF-8"?>
<root>
    <properties>
        <property name="A">a</property>
        <property name="B">b</property>
    </properties>
</root>
person bdoughan    schedule 10.06.2013
comment
Вы когда-нибудь задумывались о создании java-библиотеки с множеством полезных адаптеров? - person mat_boy; 04.12.2013
comment
Какие изменения нужно сделать, если вместо Map‹String, String› у нас Map‹String, List‹Some_Object››??? - person Anand; 03.03.2015