我有一个List<Foo>
,想要一个Multimap<String, Foo>
我们已经按Foo
函数对getId()
进行了分组。
我正在使用Java 8,它几乎可以做到:
List<Foo> foos = ...
Map<String, List<Foo>> foosById = foos.stream().collect(groupingBy(Foo::getId));
但是,我有大量需要MultiMap<String, Foo>
的代码,所以这并不能保存任何东西,我回到使用for循环来创建我的MultiMap。我错过了一种很好的“功能性”方式吗?
答案 0 :(得分:83)
您可以使用Guava Multimaps工厂:
Multimaps.index(foos, Foo::getId);
或使用Collector接口封装对Multimaps.index的调用(如下所示,在未经优化的天真实现中)。
Multimap<String, Foo> collect = foos.stream().collect(MultimapCollector.toMultimap(Foo::getId));
和收藏家:
public class MultimapCollector<T, K, V> implements Collector<T, Multimap<K, V>, Multimap<K, V>> {
private final Function<T, K> keyGetter;
private final Function<T, V> valueGetter;
public MultimapCollector(Function<T, K> keyGetter, Function<T, V> valueGetter) {
this.keyGetter = keyGetter;
this.valueGetter = valueGetter;
}
public static <T, K, V> MultimapCollector<T, K, V> toMultimap(Function<T, K> keyGetter, Function<T, V> valueGetter) {
return new MultimapCollector<>(keyGetter, valueGetter);
}
public static <T, K, V> MultimapCollector<T, K, T> toMultimap(Function<T, K> keyGetter) {
return new MultimapCollector<>(keyGetter, v -> v);
}
@Override
public Supplier<Multimap<K, V>> supplier() {
return ArrayListMultimap::create;
}
@Override
public BiConsumer<Multimap<K, V>, T> accumulator() {
return (map, element) -> map.put(keyGetter.apply(element), valueGetter.apply(element));
}
@Override
public BinaryOperator<Multimap<K, V>> combiner() {
return (map1, map2) -> {
map1.putAll(map2);
return map1;
};
}
@Override
public Function<Multimap<K, V>, Multimap<K, V>> finisher() {
return map -> map;
}
@Override
public Set<Characteristics> characteristics() {
return ImmutableSet.of(Characteristics.IDENTITY_FINISH);
}
}
答案 1 :(得分:1)
Guava 21.0 引入了几种返回 Collector
实例的方法,这些实例会将 Stream
转换为 Multimap
,这些实例根据将函数应用于其元素的结果分组。这些方法是:
ImmutableListMultimap.toImmutableListMultimap
— 创建一个 ImmutableListMultimap
ImmutableSetMultimap.toImmutableSetMultimap
— 创建一个 ImmutableSetMultimap
Multimaps.toMultimap
— 使用给定的 Supplier
Multimap
ImmutableListMultimap<String, Foo> foosById = foos.stream().collect(
ImmutableListMultimap.toImmutableListMultimap(
Foo::getId, Function.identity()));
ImmutableSetMultimap<String, Foo> foosById = foos.stream().collect(
ImmutableSetMultimap.toImmutableSetMultimap(
Foo::getId, Function.identity()));
HashMultimap<String, Foo> foosById = foos.stream().collect(
Multimaps.toMultimap(
Foo::getId, Function.identity(), HashMultimap::create)
);