当前位置: 首页 > 面试题库 >

向流中添加两个Java 8流或一个额外的元素

欧盛
2023-03-14
问题内容

我可以添加流或其他元素,如下所示:

Stream stream = Stream.concat(stream1, Stream.concat(stream2, Stream.of(element));

我可以随时添加新内容,例如:

Stream stream = Stream.concat(
                       Stream.concat(
                              stream1.filter(x -> x!=0), stream2)
                              .filter(x -> x!=1),
                                  Stream.of(element))
                                  .filter(x -> x!=2);

但这是丑陋的,因为concat是静态的。如果concat是实例方法,则上面的示例将更容易阅读:

 Stream stream = stream1.concat(stream2).concat(element);

 Stream stream = stream1
                 .filter(x -> x!=0)
                 .concat(stream2)
                 .filter(x -> x!=1)
                 .concat(element)
                 .filter(x -> x!=2);

我的问题是:

1)是否有充分的理由为什么它concat是静态的?还是我缺少一些等效的实例方法?

2)无论如何,有没有更好的方法呢?


问题答案:

如果为 Stream.concatStream.of 添加 静态导入 ,则第一个示例可以编写如下:

Stream<Foo> stream = concat(stream1, concat(stream2, of(element)));

导入具有通用名称的 静态方法 可能会导致代码变得难以阅读和维护( 命名空间污染 )。因此,最好使用更有意义的名称创建自己的 静态方法
。但是,为了演示起见,我将坚持使用该名称。

public static <T> Stream<T> concat(Stream<? extends T> lhs, Stream<? extends T> rhs) {
    return Stream.concat(lhs, rhs);
}
public static <T> Stream<T> concat(Stream<? extends T> lhs, T rhs) {
    return Stream.concat(lhs, Stream.of(rhs));
}

使用这两种静态方法(可以选择与静态导入结合使用),可以将两个示例编写如下:

Stream<Foo> stream = concat(stream1, concat(stream2, element));

Stream<Foo> stream = concat(
                         concat(stream1.filter(x -> x!=0), stream2).filter(x -> x!=1),
                         element)
                     .filter(x -> x!=2);

现在,该代码明显缩短了。但是,我同意可读性并没有提高。所以我有另一个解决方案。

在许多情况下,可以使用 收集器扩展 流的功能。将两个 收集器 放在底部,可以将两个示例编写如下:

Stream<Foo> stream = stream1.collect(concat(stream2)).collect(concat(element));

Stream<Foo> stream = stream1
                     .filter(x -> x!=0)
                     .collect(concat(stream2))
                     .filter(x -> x!=1)
                     .collect(concat(element))
                     .filter(x -> x!=2);

所需语法和上面语法之间的唯一区别是,您必须用 collect(concat(…)) 替换 concat(…
。这两种静态方法可以按以下方式实现(可以选择与静态导入结合使用):

private static <T,A,R,S> Collector<T,?,S> combine(Collector<T,A,R> collector, Function<? super R, ? extends S> function) {
    return Collector.of(
        collector.supplier(),
        collector.accumulator(),
        collector.combiner(),
        collector.finisher().andThen(function));
}
public static <T> Collector<T,?,Stream<T>> concat(Stream<? extends T> other) {
    return combine(Collectors.toList(),
        list -> Stream.concat(list.stream(), other));
}
public static <T> Collector<T,?,Stream<T>> concat(T element) {
    return concat(Stream.of(element));
}

当然,这种解决方案有一个缺点,应该提到。 collect 是消耗流中所有元素的最终操作。最重要的是,每次在链中使用收集器 concat时,
都会创建一个中间 ArrayList 。两种操作都会对程序的行为产生重大影响。但是,如果 可读性性能
更重要,它可能仍然是一个非常有用的方法



 类似资料:
  • 我可以添加流或额外的元素,像这样: 我可以边走边添加新的东西,比如: 但这很难看,因为是静态的。如果是一个实例方法,那么上面的示例将更容易阅读: 而且 2)无论如何,有没有更好的方法?

  • 如何以另一种方式从流中添加两个不同列表中的对象? 这里有一个例子

  • 正如这篇stackoverflow文章中所述,在调用终端操作之前,不会实际应用筛选器。既然我在调用终端操作之前重新分配了stream的值,那么上面的代码是否仍然是使用Java8流的适当方法呢?

  • 我有两个数据流和。

  • 问题内容: 2个流: 给定可读流, 并且 获取包含 并 连接 流的惯用(简洁)方法是什么? 我不能做,因为这样流内容混杂在一起。 n个 流: 给定一个EventEmitter发出不确定数量的流,例如 一种将 所有流串联在一起的流 的惯用(简洁)方法是什么? 问题答案: 该合并的流包会连接流。自述文件中的示例: 我相信您必须立即添加所有流。如果队列为空,则自动结束。参见问题5。 该流流库是一个具有明

  • 我想创建,它是和元素的和,换句话说,。 我该如何使用流操作来执行此操作呢?