To reuse a stream

Brian Goetz brian.goetz at oracle.com
Fri Feb 22 13:01:22 PST 2013


The fundamental question here is: whether a Stream is more like an 
Iterable, or an Iterator.  And it has been definitively answered -- the 
latter.

We flirted with a design that put Stream methods on Iterable, and it had 
all sorts of problems, many of which come down to: a Stream has no clue 
what its source is or whether it is repeatable -- and shouldn't.  (Note 
that the specification for iterator() in Iterable isn't even clear on 
this either.)

The word "stream" indicates a flow; the values flow by, and once a value 
has gone by, its gone.

If your source is repeatable (say, an immutable collection), then you 
don't need a new abstraction -- just keep calling c.stream() on your 
collection.

More generally, you can always use Supplier to add the level of 
indirection you seek:

   Supplier<IntStream> streamMaker
       = () -> Streams.intRange(2, MAX_VALUE);

And then you can call streamMaker.get() as many times as you like. 
There's no need for a StreamMaker / StreamSource / Streamable abstraction.


On 2/22/2013 3:23 PM, Ricky Clarkson wrote:
> Hi,
>
> If I run the following:
>
> IntStream numbers = Streams.intRange(2, Integer.MAX_VALUE);
> IntStream primes = numbers.filter(x -> x < 3 ||
> numbers.limit(x).noneMatch(y -> x % y == 0));
> primes.limit(100).forEach(System.out::println);
>
> I get an IllegalStateException at the forEach call: Stream is already
> linked to a child stream, which was a surprise.  I guessed that the problem
> is that I'm using the same stream twice recursively.  The following works
> fine:
>
> IntStream numbers = Streams.intRange(2, Integer.MAX_VALUE);
> IntStream primes = numbers.filter(x -> x < 3 || Streams.intRange(2, x -
> 1).noneMatch(y -> x % y == 0));
> primes.limit(100).forEach(System.out::println);
>
> This seems an odd design, I wouldn't naturally expect such a Stream to have
> a problem with concurrent/recursive use.  It means that a Stream is an
> object I need to protect/manage the ownership of, like an Iterator, and not
> so much like an Iterable, even if the underlying data source is immutable.
>   Is this deliberate?  If so will there be another abstraction that can be
> used and shared more freely?
>
> Ricky.
>


More information about the lambda-dev mailing list