我需要以Java 8 merge之类的方式在如下所示的循环中理想地合并n个哈希图:

地图1:{Name:XXX,Phn:123,Work:""}
地图2:{Name:XXX,Phn:456,Work: xyz}
地图3:{Name:XXX,Phn:789,Work:""}
我想得到如下输出:

{
   Name:XXX,        // if all values for a key are same take one
   Phn:123/456/789  // merge all non null values for same key
   Work:xyz         // make sure non-null values are never replaced
}

当我尝试像这样使用putall
public Map<String,String> mergeOriginalDataMaps(List<Integer> indexList, List<Map<String,String>> originalData) {
    Map<String,String> tmpMap = new HashMap<String,String> ();
    for (int index : indexList ) {
        tmpMap.putAll(originalData.get(index));
    }
    return tmpMap;
}

如果重复键的值为"",则先前的值将替换为新的值。我需要连接值而不是替换它们。

最佳答案

如果您有一个List<Map<String, String>>,它表示要合并的地图列表,则可以

Map<String, String> result =
    maps.stream()
        .flatMap(m -> m.entrySet().stream())
        .collect(Collectors.groupingBy(
            Map.Entry::getKey,
            Collectors.mapping(
                Map.Entry::getValue,
                Collectors.collectingAndThen(Collectors.<String>toSet(), s -> String.join("", s))
            )
        ));

此平面将每个地图映射到其条目的流。然后按每个条目的值对Stream进行分组,并将具有相同键的所有不同元素映射到其值,并进行连接。

通过首先收集Set内的所有值,可以使不同部分成为可能。

样例代码:
public static void main(String[] args) {
    List<Map<String, String>> maps = new ArrayList<>();
    maps.add(new HashMap<String, String>(){{ put("Name", "XXX"); put("Phn", "123"); put("Work", ""); }});
    maps.add(new HashMap<String, String>(){{ put("Name", "XXX"); put("Phn", "456"); put("Work", "xyz"); }});
    maps.add(new HashMap<String, String>(){{ put("Name", "XXX"); put("Phn", "789"); put("Work", ""); }});

    Map<String, String> result =
        maps.stream()
            .flatMap(m -> m.entrySet().stream())
            .collect(Collectors.groupingBy(
                Map.Entry::getKey,
                Collectors.mapping(
                    Map.Entry::getValue,
                    Collectors.collectingAndThen(Collectors.<String>toSet(), s -> String.join("", s))
                )
            ));

    System.out.println(result); // prints "{Phn=123456789, Work=xyz, Name=XXX}"
}

07-27 20:16