I use @XmlJavaTypeAdapter
to transform Map<String, MapItem>
objects into List<MapItem>
when marshalling (and vice versa when unmarshalling.)
The list always has a surrounding @XmlElement
and I want to get rid of it as it clutters the resulting XML.
How can this be done?
Or, in other words, how can I get rid of element map
in the following XML:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<top>
<map>
<item val="some-val" key="some-key"/>
</map>
</top>
MapItem
is a simple class with a key and a value:
public static class MapItem {
@XmlAttribute(name = "key")
String key;
@XmlAttribute(name = "val")
String val;
}
Map<String, MapItem>
either implicitly or explicitly contains an @XmlElement
annotation:
@XmlJavaTypeAdapter(MyMapItemAdapter.class)
@XmlElement(name = "map")
Map<String, MapItem> map = new TreeMap<String, MapItem>();
@XmlJavaTypeAdapter
:
@XmlType(name = "map-type", propOrder = { "list" })
static class MyMapItemType {
@XmlElement(name = "item")
List<MapItem> list = new ArrayList<MapItem>();
}
@XmlTransient
static final class MyMapItemAdapter extends
XmlAdapter<MyMapItemType, Map<String, MapItem>> {
MyMapItemAdapter() {
}
@Override
public MyMapItemType marshal(Map<String, MapItem> arg0)
throws Exception {
MyMapItemType myMapType = new MyMapItemType();
for (Entry<String, MapItem> entry : arg0.entrySet()) {
myMapType.list.add(entry.getValue());
}
return myMapType;
}
@Override
public Map<String, MapItem> unmarshal(MyMapItemType arg0)
throws Exception {
TreeMap<String, MapItem> treeMap = new TreeMap<String, MapItem>();
for (MapItem myEntryType : arg0.list) {
treeMap.put(myEntryType.key, myEntryType);
}
return treeMap;
}
}
@XmlRootElement(name = "top")
@XmlType(name = "top")
@XmlAccessorType(XmlAccessType.FIELD)
public class JaxbMapTest {
package xml;
import java.io.File;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;
import java.util.Map.Entry;
import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.XmlTransient;
import javax.xml.bind.annotation.XmlType;
import javax.xml.bind.annotation.adapters.XmlAdapter;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
@XmlRootElement(name = "top")
@XmlType(name = "top")
@XmlAccessorType(XmlAccessType.FIELD)
public class JaxbMapTest {
public static class MapItem {
@XmlAttribute(name = "key")
String key;
@XmlAttribute(name = "val")
String val;
}
@XmlTransient
static final class MyMapItemAdapter extends
XmlAdapter<MyMapItemType, Map<String, MapItem>> {
MyMapItemAdapter() {
}
@Override
public MyMapItemType marshal(Map<String, MapItem> arg0)
throws Exception {
MyMapItemType myMapType = new MyMapItemType();
for (Entry<String, MapItem> entry : arg0.entrySet()) {
myMapType.list.add(entry.getValue());
}
return myMapType;
}
@Override
public Map<String, MapItem> unmarshal(MyMapItemType arg0)
throws Exception {
TreeMap<String, MapItem> treeMap = new TreeMap<String, MapItem>();
for (MapItem myEntryType : arg0.list) {
treeMap.put(myEntryType.key, myEntryType);
}
return treeMap;
}
}
@XmlType(name = "map-type", propOrder = { "list" })
static class MyMapItemType {
@XmlElement(name = "item")
List<MapItem> list = new ArrayList<MapItem>();
}
public static void main(String[] args) {
try {
// Setup object
JaxbMapTest jaxbMapTest = new JaxbMapTest();
MapItem mapItem = new MapItem();
mapItem.key = "some-key";
mapItem.val = "some-val";
jaxbMapTest.add(mapItem);
// Marshal
JAXBContext jaxbContext = JAXBContext
.newInstance(JaxbMapTest.class);
Marshaller marshaller = jaxbContext.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT,
Boolean.TRUE);
marshaller.marshal(jaxbMapTest, new File("JaxbMapTest.out"));
// Exit
System.exit(0);
} catch (Exception e) {
e.printStackTrace();
System.exit(1);
}
}
@XmlJavaTypeAdapter(MyMapItemAdapter.class)
@XmlElement(name = "map")
Map<String, MapItem> map = new TreeMap<String, MapItem>();
void add(MapItem mapItem) {
map.put(mapItem.key, mapItem);
}
}
Note: I'm the EclipseLink JAXB (MOXy) lead and a member of the JAXB (JSR-222) expert group.
You could use the @XmlPath(".")
extension in MOXy to map this use case. Specifying "."
as the path indicates that the contents of the child should be written into the parent objects element.
import java.util.HashMap;
import javax.xml.bind.annotation.*;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
import org.eclipse.persistence.oxm.annotations.XmlPath;
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Foo {
@XmlJavaTypeAdapter(MyMapItemAdapter.class)
@XmlPath(".")
Map<String, MapItem> map = new TreeMap<String, MapItem>();
}
Full Example
For More Information