混合两个元素的 Java 8 Stream

标签 java java-8 java-stream

我在数组列表中有很多 Slot 类型的对象。

Slot类如下图-

Slot{
   int start;
   int end;
}

让类型列表List<Slot>被称为slots .插槽根据开始时间排序。一个时隙的结束时间可能等于下一个时隙的开始时间,但它们永远不会重叠。

有什么方法可以让我使用 Java 8 流遍历此列表,如果一个槽的结束时间与下一个槽的开始时间匹配,则合并两个槽并将它们输出到 ArrayList 中?

最佳答案

我的免费 StreamEx 完美支持这种场景增强标准 Stream API 的库。有一个 intervalMap能够将多个相邻流元素折叠为单个元素的中间操作。这是完整的示例:

// Slot class and sample data are taken from @Andreas answer
List<Slot> slots = Arrays.asList(new Slot(3, 5), new Slot(5, 7), 
                new Slot(8, 10), new Slot(10, 11), new Slot(11, 13));

List<Slot> result = StreamEx.of(slots)
        .intervalMap((s1, s2) -> s1.end == s2.start,
                     (s1, s2) -> new Slot(s1.start, s2.end))
        .toList();
System.out.println(result);
// Output: [3-7, 8-13]

intervalMap 方法有两个参数。第一个是 BiPredicate,它从输入流中接受两个相邻的元素,如果它们必须合并则返回 true(这里的条件是 s1.end == s2.start)。第二个参数是一个 BiFunction,它从合并的系列中获取第一个和最后一个元素并生成结果元素。

请注意,例如,如果您有 100 个应该合并为一个的相邻插槽,则此解决方案不会创建 100 个中间对象(就像@Misha 的回答一样,这仍然非常有趣),它会跟踪第一个和最后一个插槽该系列立即忘记了中间的一次。当然这个解决方案是并行友好的。如果您有数千个输入槽,使用 .parallel() 可能会提高性能。

请注意,当前的实现将重新创建 Slot,即使它没有与任何东西合并。在这种情况下,BinaryOperator 两次接收相同的 Slot 参数。如果你想优化这种情况,你可以进行额外的检查,比如 s1 == s2 ? s1 : ...:

List<Slot> result = StreamEx.of(slots)
        .intervalMap((s1, s2) -> s1.end == s2.start,
                     (s1, s2) -> s1 == s2 ? s1 : new Slot(s1.start, s2.end))
        .toList();

关于混合两个元素的 Java 8 Stream,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32771415/

相关文章:

java - hibernate什么时候会更新相关实体?

java - 通过对值求和将 map 列表合并为单个 map

java - 策略游戏帮助 : fog of war

java - 从 SynchronizedMultimap 中删除元素会增加意外行为

java - jaxb2-maven-plugin :schemagen to work with JDK8? 是否有修复

java - 如果流过滤条件不返回任何结果,则返回单个元素的列表

Java 8 Streams 中间操作会产生新的流吗?

java - Peek() 真正地看到元素流过管道中的某个点

java - 如何获得每个项目的最新 10 个访问者

java - 无法使用 FileReader 找到文件