Java 9 Collectors.flatMapping在Java 8中重写

Nik*_*las 12 java java-8 java-stream collectors java-9

我接触到了一个新功能,因为被称为Collectors.flatMapping分组或分区的下游.如(例子来自这里):

List<List<Integer>> list = Arrays.asList(
    Arrays.asList(1, 2, 3, 4, 5, 6), 
    Arrays.asList(7, 8, 9, 10));

Map<Integer, List<Integer>> map =list.stream()
    .collect(Collectors.groupingBy(
         Collection::size,
         Collectors.flatMapping(
             l -> l.stream().filter(i -> i % 2 == 0),
             Collectors.toList())));
Run Code Online (Sandbox Code Playgroud)

{4 = [8,10],6 = [2,4,6]}

这是一个相当优雅的方式,只使用3个收藏家.我需要在中重写收集器,但尚不支持.我尝试使用6种收集器,这种收集器使用起来 非常广泛,我无法找到使用较少的收集器的方法:

Map<Integer, List<Integer>> map = list.stream()
    .collect(Collectors.groupingBy(
        Collection::size,
        Collectors.collectingAndThen(
            Collectors.mapping(
                l -> l.stream().filter(i -> i % 2 == 0).collect(Collectors.toList()),
                Collectors.toList()),
            i -> i.stream().flatMap(j -> j.stream()).collect(Collectors.toList()))));
Run Code Online (Sandbox Code Playgroud)

是否有一个更短的仅使用更好的方法

Mic*_*ael 14

我只是向后移动flatMapping.它只需要2个方法和1个类,没有其他依赖项.

此外,当需要升级到Java 9时,您可以弃用您的版本并使用正确的版本替换它的任何用法.

以下代码取自JDK.我没有写.我用你的例子测试了它,它返回相同的结果.

如果你想使用它,Holger写了一个较短的版本.我会相信它,但我没有测试过它.

class Nikollectors
{
   public static <T, U, A, R> Collector<T, ?, R> flatMapping(Function<? super T, ? extends Stream<? extends U>> mapper, Collector<? super U, A, R> downstream) {
        BiConsumer<A, ? super U> downstreamAccumulator = downstream.accumulator();
        return new CollectorImpl<>(downstream.supplier(),
            (r, t) -> {
                try (Stream<? extends U> result = mapper.apply(t)) {
                    if (result != null)
                        result.sequential().forEach(u -> downstreamAccumulator.accept(r, u));
                }
            },
            downstream.combiner(), downstream.finisher(),
            downstream.characteristics());
    }

   private static class CollectorImpl<T, A, R> implements Collector<T, A, R>
   {
        private final Supplier<A> supplier;
        private final BiConsumer<A, T> accumulator;
        private final BinaryOperator<A> combiner;
        private final Function<A, R> finisher;
        private final Set<Characteristics> characteristics;

        CollectorImpl(Supplier<A> supplier,
                      BiConsumer<A, T> accumulator,
                      BinaryOperator<A> combiner,
                      Function<A,R> finisher,
                      Set<Characteristics> characteristics) {
            this.supplier = supplier;
            this.accumulator = accumulator;
            this.combiner = combiner;
            this.finisher = finisher;
            this.characteristics = characteristics;
        }

        CollectorImpl(Supplier<A> supplier,
                      BiConsumer<A, T> accumulator,
                      BinaryOperator<A> combiner,
                      Set<Characteristics> characteristics) {
            this(supplier, accumulator, combiner, castingIdentity(), characteristics);
        }

        @Override
        public BiConsumer<A, T> accumulator() {
            return accumulator;
        }

        @Override
        public Supplier<A> supplier() {
            return supplier;
        }

        @Override
        public BinaryOperator<A> combiner() {
            return combiner;
        }

        @Override
        public Function<A, R> finisher() {
            return finisher;
        }

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

   private static <I, R> Function<I, R> castingIdentity() {
       return i -> (R) i;
   }
}
Run Code Online (Sandbox Code Playgroud)

样品用法:

Map<Integer, List<Integer>> map =list.stream()
    .collect(Collectors.groupingBy(
         Collection::size,
         Nikollectors.flatMapping( // <<<
             l -> l.stream().filter(i -> i % 2 == 0),
             Collectors.toList()
        )
    )
);
Run Code Online (Sandbox Code Playgroud)

  • 你可能在开玩笑,但在我的项目中,我们有一个包`xxx.HolgerUtil` - 只是为了确定这是从哪里取来的,不是开玩笑.:) (6认同)
  • 你应该得到两个"Nikollectors"的upvotes ...你应该添加一个使用示例,并且可能是一个静态工厂方法,它会返回这个`Collectors :: flatMapping`,使其成为一个完整的例子恕我直言 (5认同)
  • @Flown Holger已经完成了,我正在链接到.关键不仅仅是展示任何旧的实现,而是展示(尽可能)我在首先需要*任何*未来JDK功能时首先采用的方法.这将涉及复制粘贴,而不是自己从头开始编写.简洁对我来说并不重要,因为我会把它放到一个util类中,可能永远不会再看它了. (3认同)
  • @Michael刚刚进行复制和粘贴可能会很安静,因此我们建议您仅在完全理解它时仍然使用它.与我的版本的主要区别在于,JDK内部代码使用`CollectorImpl`类,因为最后,必须有一个`Collector`接口的实际实现类.在您的应用程序代码中,您可以使用`Collector.of(...)`代替它,它将使用JRE提供的非公共实现类. (3认同)
  • 如果你没有复制`CollectorImpl`,你就不需要"轻易地"区分它.你复制的越多,就越难.这些工件可能经历任意的JRE重构,与被移植的`flatMapping`方法完全无关,比如被移出这个类或重命名等等.你没有声称已经写过它,但是你刚刚在Stackoverflow上发布它,这意味着您刚刚根据[服务条款](https://stackoverflow.com/legal/terms-of-service/public)授予Stackoverflow CC-BY-SA许可,这是您最好只做的一个动作有权这样做...... (3认同)
  • 你们两个都是我的一天.:D感谢您的答案和有用的链接. (2认同)

Eug*_*ene 9

对于这种特殊情况,我想这将是一个更简单的版本:

Map<Integer, List<Integer>> map =
        list.stream()
            .collect(Collectors.toMap(
                Collection::size,
                x -> x.stream().filter(y -> y % 2 == 0).collect(Collectors.toList())
            ));
Run Code Online (Sandbox Code Playgroud)

如果涉及合并(两个集合具有相同的大小),我会添加一个merge非常简单的函数:

 Map<Integer, List<Integer>> map =
        list.stream()
            .collect(Collectors.toMap(
                Collection::size,
                x -> x.stream().filter(y -> y % 2 == 0).collect(Collectors.toCollection(ArrayList::new)),
                (left, right) -> {
                    left.addAll(right);
                    return left;
                }
            ));
Run Code Online (Sandbox Code Playgroud)

否则,我同意迈克尔在这个评论中,这并不难以反向移植到java-8.