通过项目的属性索引Collection的最简洁方法,该项目本身就是一个集合

时间:2014-04-11 04:51:47

标签: java guava java-8 java-stream

我有一个List<Foo>,想要一个番石榴Multimap<String, Foo>我们已根据Foo函数的每个标记对Collection<String> getTags()进行了分组。

我正在使用java 8,所以lambdas和方法引用很好/鼓励。

例如,如果我有:

foo1, tags=a,b,c
foo2, tags=c,d
foo3, tags=a,c,e

我会得到一个Multimap<String, Foo>

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

4 个答案:

答案 0 :(得分:12)

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

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();

这不会造成额外的副作用(请参阅here),并发且更具惯用性。

您还可以将这些特殊的lambda提取到一个完整的收集器中,如下所示:

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)方法返回characteristics()。这可以使内部收集机器更有效地发挥作用,特别是在并行减少的情况下。

答案 1 :(得分:8)

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

答案 2 :(得分:1)

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

ImmutableSetMultimap<String, Foo> result = list.stream()
        .flatMap(foo -> foo.getTags().stream().map(tag -> Map.entry(tag, foo)))
        .collect(ImmutableSetMultimap.toImmutableSetMultimap(Map.Entry::getKey, Map.Entry::getValue));

答案 3 :(得分:0)

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

    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收集器。其他用例也有一个Multmaps.flatteningToMultimap