java - 如何使用Xstream的addImplicitMap方法?

标签 java xml xstream

我想借助 xstream 中可用的 addImplicitMap 方法来序列化该类。类将如下所示:

class MapTest{
private Map<String, String> mapList;

public MapTest() {

    mapList= new HashMap<String, String>();
}

public void setServicesHealth(String id, String name) {
    map.put(id, name);
}

我尝试过:

class MapTestMain{
public static void main(String args[]){ 
MapTest services = new MapTest();
services.setServicesHealth("ID01", "Jack");
services.setServicesHealth("ID02", "Neil);

    XStream stream = new XStream(new StaxDriver());
    stream.alias("MapTest", MapTest.class);
    stream.addImplicitMap(MapTest.class, "map", "id", String.class, "name");
    String xmlStr = stream.toXML(services);
    System.out.println(xmlStr);
  }
}

但是我没有得到正确的输出。我的预期输出是这样的:

<?xml version="1.0" ?>
<MapTest>
 <id>Started</id>
 <name>Started</name>
</MapTest>

请帮帮我...

最佳答案

TLDR:addImplicitMap不允许您删除 <mapList>容器元素和 entry元素并一次性重命名键/值元素(示例 1 和 2)。要更好地控制 map 中元素的命名,请使用 NamedMapConverter (示例 3)。为了能够一次完成所有操作,您将需要一个自定义 Converter实现(示例 4)。

<小时/>

长答案:

对于 addImplicitMap 似乎没有很好的解释确实如此,而且他们的文档中也不是很清楚,所以我将尝试在这里解释。

addImplicitMap将删除容器元素,但是,当像这样使用时,它不会让您删除 <entry>元素或重命名键/值元素。例如:(1)

stream.addImplicitMap(MapTest.class, "mapList", "s", Map.Entry.class, null);

结果:

<MapTest>
  <s>
    <string>ID01</string>
    <string>Jack</string>
  </s>
  <s>
    <string>ID02</string>
    <string>Neil</string>
  </s>
</MapTest>

或者,addImplicitMap让您在写入时仅存储映射的值(忽略容器元素和键元素)。读取 XML 时,它会使用值对象的指定字段重新创建映射键,因此映射键必须存储在映射值对象中。

例如,我们可以将 map 的值设置为 Service具有 ID 属性的对象,并使 Xstream 仅存储值并从中重新创建 map :(2)

private Map<String, Service> serviceMap; // modified in MapTest

static class Service {
  private String id, name;

  public Service(String id, String name) {
    this.id = id;
    this.name = name;
  }
}

MapTest services = new MapTest();
services.setService("ID01", new Service("ID01", "Jack"));
services.setService("ID02", new Service("ID02", "Neil"));

stream.addImplicitMap(MapTest.class, "serviceMap", "s", Service.class, "id");

输出:

<MapTest>
  <s>
    <id>ID01</id>
    <name>Jack</name>
  </s>
  <s>
    <id>ID02</id>
    <name>Neil</name>
  </s>
</MapTest>

请注意,这里每个 <s>元素实际上是序列化的Service对象,而不是 Map.Entry对象。

<小时/>

这仍然不能完全解决您的问题(我们也必须更改 map 中的内容),因此我们可以尝试 namedMapConverter 。如果我们回到使用 <String, String> 的 map 正如您最初那样,我们可以使用:(3)

stream.registerConverter(new NamedMapConverter(stream.getMapper(),
                null, "id", String.class, "name", String.class));

这给出了输出:

<MapTest>
  <serviceMap>
    <id>ID01</id>
    <name>Jack</name>
    <id>ID02</id>
    <name>Neil</name>
  </serviceMap>
</MapTest>

仍然不太正确,我不相信如果不实现自定义 Converter 就可以让它变得更好类(有一个很好的教程 here )。所以我们添加这一行(而不是 NamedMapConverter ):

stream.registerConverter(new MapTestConverter());

并使用自定义转换器实现:(4)

static class MapTestConverter implements Converter {        
    public boolean canConvert(Class type) {
        return type.equals(MapTest.class);
    }

    public void marshal(Object source, HierarchicalStreamWriter writer,
            MarshallingContext context) {
        MapTest mt = (MapTest) source;
        for (Entry<String, String> e : mt.serviceMap.entrySet()) {
            writer.startNode("id");
            writer.setValue(e.getKey());
            writer.endNode();
            writer.startNode("name");
            writer.setValue(e.getValue());
            writer.endNode();
        }
    }

    public Object unmarshal(HierarchicalStreamReader reader, UnmarshallingContext context) {
        MapTest mt = new MapTest();
        String id = null;

        while (reader.hasMoreChildren()) {
            reader.moveDown();
            if ("id".equals(reader.getNodeName())) {
                if (id != null) { throw new RuntimeException("Malformed XML, ID was set twice: " + id); }
                id = (String) context.convertAnother(mt, String.class);
            } else if ("name".equals(reader.getNodeName())) {
                String name = (String) context.convertAnother(mt, String.class);
                if (id == null) { throw new RuntimeException("Malformed XML: Found name without ID: " + name); }
                mt.serviceMap.put(id, name);
                id = null;
            }
            reader.moveUp();
        }
        return mt;
    }
}

最终我们得到了想要的结果:

<MapTest>
  <id>ID01</id>
  <name>Jack</name>
  <id>ID02</id>
  <name>Neil</name>
</MapTest>

(抱歉,我没有足够的代表将以下内容制作为链接)

addImplicitMap 的完整代码示例:pastebin.com/MYiAde3m

namedMapConverter 的完整代码示例:pastebin.com/kVChup5x

Converter 的完整代码示例:pastebin.com/vUTwaHkk

关于java - 如何使用Xstream的addImplicitMap方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29466312/

相关文章:

java - xstream:将对象映射到 xml(不包括一个字段)

java - 如何使用serialize Collections.synchronizedList和co

xml - XSLT 中无用的 match=text()

java - 使用 XStream 1.4.2 反序列化 transient 字段

java - 在 tableView 中显示列

java - 偶尔出现 'EJBTransactionRolledbackException: Transaction rolled back'错误挂起应用

java - 具有顶部和底部边距的自定义按钮

xml - Qt 5 在 XML 中生成随机属性顺序

java - InterruptedException 未被捕获

java - 我想在这个 block 中转换为java 8流?