如何将集合转换为按嵌套集合属性的元素分组的Guava Multimap?

问题描述 投票:12回答:4

我有一个List<Foo>,想要一个番石榴Multimap<String, Foo>,我们在其中将Multimap<String, Foo>按其Foo功能的每个标签分组。

我正在使用Java 8,因此可以很好地鼓励使用lambda和方法引用。

例如,如果我有:

Collection<String> getTags()

我会得到一个foo1, tags=a,b,c foo2, tags=c,d foo3, tags=a,c,e

Multimap<String, Foo>
java guava java-8 java-stream
4个回答
12
投票

您可以为此使用自定义收集器:

a -> foo1, foo3
b -> foo1
c -> foo1, foo2, foo3
d -> foo2
e -> foo3

这不会引起额外的副作用(请参见此Multimap<String, Foo> map = list.stream().collect( ImmutableMultimap::builder, (builder, value) -> value.getTags().forEach(tag -> builder.put(tag, value)), (builder1, builder2) -> builder1.putAll(builder2.build()) ).build(); ),是同时发生的,并且更惯用。

您还可以将这些临时lambda提取到成熟的收集器中,如下所示:

here

然后该集合将如下所示:

public static <T, K> Collector<T, ?, Multimap<K, T>> toMultimapByKey(Function<? super T, ? extends Iterable<? extends K>> keysMapper) {
    return new MultimapCollector<>(keysMapper);
}

private static class MultimapCollector<T, K> implements Collector<T, ImmutableMultimap.Builder<K, T>, Multimap<K, T>> {
    private final Function<? super T, ? extends Iterable<? extends K>> keysMapper;

    private MultimapCollector(Function<? super T, ? extends Iterable<? extends K>> keysMapper) {
        this.keysMapper = keysMapper;
    }

    @Override
    public Supplier<ImmutableMultimap.Builder<K, T>> supplier() {
        return ImmutableMultimap::builder;
    }

    @Override
    public BiConsumer<ImmutableMultimap.Builder<K, T>, T> accumulator() {
        return (builder, value) -> keysMapper.apply(value).forEach(k -> builder.put(k, value));
    }

    @Override
    public BinaryOperator<ImmutableMultimap.Builder<K, T>> combiner() {
        return (b1, b2) -> b1.putAll(b2.build());
    }

    @Override
    public Function<ImmutableMultimap.Builder<K, T>, Multimap<K, T>> finisher() {
        return ImmutableMultimap.Builder<K, T>::build;
    }

    @Override
    public Set<Characteristics> characteristics() {
        return Collections.emptySet();
    }
}

如果顺序对您不重要,您也可以从Multimap<String, Foo> map = list.stream().collect(toMultimapByKey(Foo::getTags)); 方法返回EnumSet.of(Characteristics.UNORDERED)。这可以使内部收集机制更有效地发挥作用,尤其是在并行缩减的情况下。


8
投票
characteristics()

0
投票

对于Java 8流,这有点惯用了:

ImmutableMultimap.Builder<String, Foo> builder = ImmutableMultimap.builder();
list.forEach(foo -> foo.getTags().forEach(tag -> builder.put(tag, foo));
return builder.build();

我知道对类的使用很丑陋,但是我想展示的关键是 Multimap<String, Foo> map = list.stream() //First build a stream of Pair<String, Foo> .flatMap(f -> f.tags.stream().map(s -> new AbstractMap.SimpleImmutableEntry<>(s, f))) //Then collect it up into a multimap. .collect( Multimaps.toMultimap( x -> x.getKey(), x -> x.getValue(), MultimapBuilder.hashKeys().arrayListValues()::build ) ); 收集器。其他用例也有一个Multimaps.toMultmap


0
投票

您可以使用Multmaps.flatteningToMultimap在每个标记/ foo对上创建一个流,然后将其收集到多图中。较新版本的Guava为此提供了Stream.flatMapStream.flatMap等收集器。

ImmutableSetMultimap.toImmutableSetMultimap
© www.soinside.com 2019 - 2024. All rights reserved.