迭代两个Java-8-Streams

F. *_*ler 28 java java-8 java-stream

我想将两个Java-8-Streams一起迭代,以便我在每次迭代中都有两个参数.像这样的东西,somefunction产生类似的东西Stream<Pair<A,B>>.

Stream<A> as;
Stream<B> bs;
somefunction (as, bs)
  .forEach ((a, b) -> foo (a, b));
// or something like
somefunction (as, bs)
  .forEach ((Pair<A, B> abs) -> foo (abs.left (), abs.right ()));
Run Code Online (Sandbox Code Playgroud)

我想知道,如果Java提供类似的东西,虽然Pair在Java中没有:-(如果没有像这样的API函数,是否有另一种方式同时迭代两个流?

Hol*_*ger 25

static <A, B> Stream<Pair<A, B>> zip(Stream<A> as, Stream<B> bs)
{
    Iterator<A> i=as.iterator();
    return bs.filter(x->i.hasNext()).map(b->new Pair<>(i.next(), b));
}
Run Code Online (Sandbox Code Playgroud)

这不提供并行执行,但原始zip实现也没有.

正如F.Böller指出的那样,如果bs是无限的并且as不是无效的话它就不起作用.对于适用于无限和有限流的所有可能组合的解决方案,Iterator检查hasNext方法中的两个源的中间体似乎是不可避免的¹:

static <A, B> Stream<Pair<A,B>> zip(Stream<A> as, Stream<B> bs) {
    Iterator<A> i1 = as.iterator();
    Iterator<B> i2 = bs.iterator();
    Iterable<Pair<A,B>> i=()->new Iterator<Pair<A,B>>() {
        public boolean hasNext() {
            return i1.hasNext() && i2.hasNext();
        }
        public Pair<A,B> next() {
            return new Pair<A,B>(i1.next(), i2.next());
        }
    };
    return StreamSupport.stream(i.spliterator(), false);
}
Run Code Online (Sandbox Code Playgroud)

如果您希望能够平行荏苒,你应该考虑的来源Stream.例如,您可以拉链两个ArrayList(或任何RandomAccessList)像

ArrayList<Foo> l1=new ArrayList<>();
ArrayList<Bar> l2=new ArrayList<>();
IntStream.range(0, Math.min(l1.size(), l2.size()))
         .mapToObj(i->new Pair(l1.get(i), l2.get(i)))
         . …
Run Code Online (Sandbox Code Playgroud)

¹(除非你Spliterator直接实施)

  • @F。Böller:是的,解决这个问题需要做一些不小的改变。您链接的解决方案可能有效,但我建议要小心许可问题,因为正如回答者自己所说,代码已从 beta jdk 源进行了 c&amp;p 编辑。您还应该记住,可能还有其他问题,因为代码的作者可能有理由将其从 jdk 中删除...... (2认同)