迭代两个Java-8-Streams

时间:2014-06-05 12:08:16

标签: 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 ()));

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

1 个答案:

答案 0 :(得分:23)

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

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

bs为无限且as不为Iterator,则F. Böller has pointed out不起作用。对于适用于无限和有限流的所有可能组合的解决方案,在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); } 似乎是不可避免的¹:

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))) . … ),如

Spliterator

¹(除非你直接实施{{1}})