[英]Java8 - How to convert a nested maps to list of nested maps gathered by value of inner map's key
[英]Convert XML to Map of nested maps
我有一个带有嵌套元素和重复标记的XML。 例如:
<person>
<name>Rama</name>
<age>27</age>
<gender>male</gender>
<address>
<doornumber>234</doornumber>
<street>Kanon</street>
<city>Hyderabad</city>
</address>
<qualification>
<degree>M.Sc</degree>
<specialisation>Maths</specialisation>
</qualification>
<qualification>
<degree>B.E.</degree>
<specialisation>Electrical</specialisation>
</qualification>
</person>
现在我想要一个API,它将这个XML转换成Java中的Map地图:
{name="Rama",age="27",gender="male",address={doornumber=234,street="Kanon",city="Hyderabad"},qualification=[{degree="M.Sc",specialisation="Maths"},{degree="B.E.",specialisation="Electrical"}]}
我知道我们可以使用XStream API来实现这一目标。 在这里,我只是想知道使用XStream是否有任何缺点以及是否存在任何其他更好的Java API来实现这一点。 有什么建议?
注意:这应该以通用的方式完成,即Java API应该适用于任何XML,而不仅仅适用于上述XML。
但现在已经很晚了。 但是我已经为XStream API编写了一个自定义的MapEntryConverter,它可以处理任何复杂的XML数据,无论它有多深。 即使它支持重复标记名称(将存储为ArrayList)。
XStream xStream = new XStream(new DomDriver());
xStream.registerConverter(new MapEntryConverter());
xStream.alias("xml", java.util.Map.class);
// from XML, convert back to map
Map<String, List<Object>> map = (Map<String, List<Object>>) xStream.fromXML(xmlData);
/*System.out.println("MAP: \n" + map.entrySet().toString());*/
String xml = xStream.toXML(map);
/*System.out.println("XML: \n"+xml);*/
MapEntryConverter.java
import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import com.thoughtworks.xstream.converters.Converter;
import com.thoughtworks.xstream.converters.MarshallingContext;
import com.thoughtworks.xstream.converters.UnmarshallingContext;
import com.thoughtworks.xstream.io.HierarchicalStreamReader;
import com.thoughtworks.xstream.io.HierarchicalStreamWriter;
public class MapEntryConverter implements Converter{
@SuppressWarnings("rawtypes")
public boolean canConvert(Class clazz) {
return AbstractMap.class.isAssignableFrom(clazz);
}
@SuppressWarnings({ "unchecked" })
public void marshal(Object value, HierarchicalStreamWriter writer, MarshallingContext context) {
AbstractMap<String, List<?>> map = (AbstractMap<String, List<?>>) value;
List<Map<String, ?>> list = (List<Map<String, ?>>) map.get("xml");
for( Map<String, ?> maps: list ) {
for( Entry<String, ?> entry: maps.entrySet() ) {
mapToXML(writer, entry);
}
}
}
@SuppressWarnings("unchecked")
private void mapToXML(HierarchicalStreamWriter writer, Entry<String, ?> entry) {
writer.startNode(entry.getKey());
if( entry.getValue() instanceof String ) {
writer.setValue(entry.getValue().toString());
}else if( entry.getValue() instanceof ArrayList ) {
List<?> list = (List<?>) entry.getValue();
for( Object object: list ) {
Map<String, ?> map = (Map<String, ?>) object;
for( Entry<String, ?> entryS: map.entrySet() ) {
mapToXML(writer, entryS);
}
}
}
writer.endNode();
}
public Object unmarshal(HierarchicalStreamReader reader, UnmarshallingContext context) {
Map<String, List<Object>> map = new HashMap<String, List<Object>>();
map = xmlToMap(reader, new HashMap<String, List<Object>>());
return map;
}
private Map<String, List<Object>> xmlToMap(HierarchicalStreamReader reader, Map<String, List<Object>> map) {
List<Object> list = new ArrayList<Object>();
while(reader.hasMoreChildren()) {
reader.moveDown();
if( reader.hasMoreChildren() ) {
list.add(xmlToMap(reader, new HashMap<String, List<Object>>()));
}else {
Map<String, Object> mapN = new HashMap<String, Object>();
mapN.put(reader.getNodeName(), reader.getValue());
list.add(mapN);
}
reader.moveUp();
}
map.put(reader.getNodeName(), list);
return map;
}
}
请注意,如果Map中的每个值都有子项,则会将其存储为列表。 根据我的说法,解组部分非常整洁。 我会在空闲时间打磨和简化编组部分。
输入数据应该在xml标签中提供,如下所示。
<xml>
<person>
<name>Rama</name>
<age>27</age>
<gender>male</gender>
<address>
<doornumber>234</doornumber>
<street>Kanon</street>
<city>Hyderabad</city>
</address>
<qualification>
<degree>M.Sc</degree>
<specialisation>Maths</specialisation>
</qualification>
<qualification>
<degree>B.E.</degree>
<specialisation>Electrical</specialisation>
</qualification>
</person>
</xml>
将输出与预期输出进行比较时会有一点变化。 重复的键值不会合并为数组,而是它们是单独的列表。
我使用https://github.com/douglascrockford/JSON-java这个项目一次将xml转换为json。 它很快并且完成了它的工作。
就个人而言,我会使用标准的JAXP接口 - 如果你真的想要一张地图的地图,那就是SAX,如果你想保留顺序的DOM(一棵树而不是无序的嵌套地图),并且想要更好地映射到XML信息集的API。
我认为Melih可能指的是这个解决方案。 这是使用json.org库的一些Kotlin代码:
将implementation "org.json:json:20180130"
到build.gradle
。
import org.json.JSONML
fun xmlToMap(xml: String): MutableMap<String, Any>? {
return JSONML.toJSONObject(xml).toMap()
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.