2013-06-10 11 views
7

Ich habe eine Map<String, String>.
Die erste Idee, die jeder hat, ist, es in eine List<Pair<String,String>> (Pair ist eine benutzerdefinierte Klasse) zu konvertieren.JAXB @ XmlAdapter: Karte -> List Adapter? (nur Marshall)

Ich habe ein @XmlAdapter wie folgt versucht:

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

Aber Eclipse-moxy, die JAXB impl ich, mit einem ClassCastException endete - "nicht HashMap Sammlung umwandeln kann".

Wird diese Konvertierung von JAXB unterstützt? Oder habe ich einen Dokumentationsteil übersehen, der erklärt, warum das nicht der Fall ist?

PS: Ich wollte XML wie folgt erhalten:

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

ich es bekam, musste nur eine Zwischenklasse verwenden. Ebenfalls beschrieben bei Handle NPE in XMLCompositeObjectMappingNodeValue.marshalSingleValue(XMLCompositeObjectMappingNodeValue.java:161)

Antwort

8

Anstatt die Map auf eine List Anpassung, sollten Sie es auf ein Objekt anzupassen, die eine List Eigenschaft hat.

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; 
    } 

} 

Domain Model (Root)

Unten ist ein Modellobjekt mit einer Map Eigenschaft. Die @XmlJavaTypeAdapter Annotation wird verwendet, um die XmlAdapter anzugeben.

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; 

} 

Demo

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); 
    } 

} 

input.xml/Output

<?xml version="1.0" encoding="UTF-8"?> 
<root> 
    <properties> 
     <property name="A">a</property> 
     <property name="B">b</property> 
    </properties> 
</root> 
+1

Haben Sie schon einmal darüber nachgedacht eine Java-Bibliothek mit vielen nützlichen Adapter zu schaffen? –

+0

Was ändert sich, wenn statt Map , Map > ??? – Anand