如
此答案所述,这是一个已知问题,将在Java 9中得到解决——至少对于不接受合并函数的
toMap
收集器。
由于合并函数仅接收要合并的两个值且签名不能轻易更改,因此这些重载方法没有解决方案。不幸的是,没有接受Map
Supplier
而不带显式合并函数的toMap
收集器,因此除非在发布之前发生变化,否则您需要返回LinkedHashMap
的情况将无法得到解决。
因此,解决方案是实现自己的收集器。然后,您就不必等待Java 9,并且不必冒失望的风险。
static <T, K, V, M extends Map<K, V>> Collector<T, ?, M> toMap(
Function<T, K> keyExtractor, Function<T, V> valueExtractor, Supplier<M> mapSupplier) {
return Collector.of(mapSupplier,
(m, t) -> putUnique(m, keyExtractor.apply(t), valueExtractor.apply(t)),
(m1,m2)-> { m2.forEach((k, v) -> putUnique(m1, k, v)); return m1; }
);
}
private static <K,V> void putUnique(Map<K,V> map, K key, V v1){
V v2 = map.putIfAbsent(key, v1);
if(v2 != null) throw new IllegalStateException(
String.format("Duplicate key %s (values %s and %s)", key, v1, v2));
}
您可以将此收集器用作:
LinkedHashMap<KeyType, ValueType> map = myList.stream()
.collect(toMap(MyObject::getKey, MyObject::getValue, LinkedHashMap::new))
或者使用一个合格的
MyCollector.toMap
,引用您放置自定义收集器的类。