Подтвердить что ты не робот

Самый чистый способ создания мультикана guava из потока java8

У меня есть 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. Есть ли хороший "функциональный" способ, который мне не хватает?

4b9b3361

Ответ 1

Вы можете просто использовать Guava Multimaps factory:

Multimaps.index(foos, Foo::getId);

или заверните вызов 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);
    }
}