我以为我在Java 8流方面做得很好,但后来......
我有一个Foo
界面:
public interface Foo {
String getKey();
Stream<Bar> bars();
}
我知道我可以使用每个密钥将Stream<Foo>
收集到Map<String, Foo>
中:
Map<String, Foo> foosByKey = fooStream.collect(
Collectors.toMap(Foo::getKey, Function.identity()));
但如果我想将它们收集到Map<Bar, Foo>
怎么办?换句话说,对于Steam中的每个Foo
,我想将Foo
放在地图中,键入Bar
返回的每个Foo.bars()
个实例。我从哪里开始?
答案 0 :(得分:3)
根据建议here,您需要从每个Bar
中提取Foo
值并创建它们对。完成配对后,您可以将它们收集到Map
中。例如,
Map<Bar, Foo> map = fooStream.flatMap(foo -> foo.bars().map(bar -> new SimpleEntry<>(bar, foo)))
.collect(Collectors.toMap(Entry::getKey, Entry::getValue));
我们在这里使用SimpleEntry
因为它可用(Java没有更简单的Pair
类型)。你可以写自己的更具体。
答案 1 :(得分:1)
您可以为此定义一个新的收集器。一个简单的实现(总是创建ArrayList的HashMap;没有下游支持)可以是:
public static <T, K>
Collector<T, ?, Map<K, List<T>>> multiGroupingBy(
Function<? super T, Collection<? extends K>> multiClassifier) {
return Collector.of(
HashMap::new,
(map, entry) -> {
multiClassifier.apply(entry)
.forEach(
key -> map
.computeIfAbsent(key,
__ -> new ArrayList<>())
.add(entry));
},
(map1, map2) -> {
map2.forEach(
(key, list) -> map1
.computeIfAbsent(key,
__ -> new ArrayList<>())
.addAll(list));
return map1;
});
}
然后您可以致电:
fooStream.collect(multiGroupingBy(Foo::bars));