Stream generators

Brian Goetz brian.goetz at oracle.com
Sat Dec 1 05:43:44 PST 2012


The issue of creating your own streams via merging is tricky because 
we're missing the language features (yield, laziness) that tend to make 
that easy.  You can iterate the contents of a Stream with Iterator; you 
can create a new Stream by creating an Iterator.  Writing Iterators by 
hand is unpleasant.

So I wrote a clunky "stream merger" class that was about a page of code 
to deal with the impedence mismatches; it deals with buffering elements 
from the input and output iterators, and is driven by a lambda that 
receives an extra "controller" argument that lets it control the input 
streams, as well as a peek at the current left and right values.  With 
that, writing your first example looked like:

         StreamMerger<Integer> summer
         = new StreamMerger((cursor, peekLeft, peekRight) -> {
             if (cursor.rightIsEmpty())
                 return cursor.advanceLeft();
             else if (cursor.leftIsEmpty())
                 return cursor.advanceRight();
             else
                 return cursor.advanceLeft() + cursor.advanceRight();
         });
         Stream<Integer> sum = summer.merge(leftStream, rightStream);

The stream merger creates an Iterator that consults the lambda about 
what to do.  (Alternately you could write it as a SAM abstract class, 
and implement the "next" method.)

Not quite as pretty as your examples, but serviceable.


On 11/30/2012 3:38 PM, Joe Bowbeer wrote:
> Here are two examples of methods that combine two streams:
>
> 1. An addStreams method, like the one below written in Python that
> generates a stream by summing two other streams:
>
>      def addStreams(si, sj): # add corres. elements of two streams
>          if not si: return sj
>          if not sj: return si
>          tailsum = lambda ti=si, tj=sj : addStreams(tail(ti), tail(tj))
>          return (head(si) + head(sj), tailsum)
>
> Source:
>
> http://code.activestate.com/lists/python-list/9029/
>
>
> 2. A merge method, like the one below written in Scala that generates
> Hamming numbers:
>
>    def hamming: Stream[Int] =
>      Stream.cons(1, merge(hamming.map(2*), merge(hamming.map(3*),
> hamming.map(5*))))
>
> Where the merge method is defined:
>
>    def merge[T](a: Stream[T], b: Stream[T])(implicit ord: Ordering[T]):
> Stream[T] = {
>      if (b.isEmpty)
>        a
>      else if (a.isEmpty)
>        b
>      else {
>        val order = ord.compare(a.head, b.head)
>        if (order < 0)
>          Stream.cons(a.head, merge(a.tail, b))
>        else if (order > 0)
>          Stream.cons(b.head, merge(a, b.tail))
>        else
>          Stream.cons(a.head, merge(a.tail, b.tail))
>      }
>    }
>
> Source:
>
> http://code.google.com/p/wing-ding/source/browse/trunk/books/Programming_Scala/src/adhoc/HammingNumbers/src/hamming/Main.scala
>
>
> Is it easy to write the corresponding methods in Java?
>
> Joe
>
>
>
> On Fri, Nov 30, 2012 at 10:09 AM, Joe Bowbeer <joe.bowbeer at gmail.com
> <mailto:joe.bowbeer at gmail.com>> wrote:
>
>     I mean a more general merge which may emit an element from either
>     stream, depending, and may drop some elements from one or both streams.
>
>     On Nov 30, 2012 9:51 AM, "Brian Goetz" <brian.goetz at oracle.com
>     <mailto:brian.goetz at oracle.com>> wrote:
>
>             I think it would be beneficial for comparison to show a bit
>             of their
>             implementations.
>
>
>         Here's iterate(seed, UnaryOperator):
>
>              public static<T> Stream<T> iterate(final T seed, final
>         UnaryOperator<T> f) {
>                  Objects.requireNonNull(f);
>                  final InfiniteIterator<T> iterator = new
>         InfiniteIterator<T>() {
>                      T t = null;
>
>                      @Override
>                      public T next() {
>                          return t = (t == null) ? seed : f.operate(t);
>                      }
>                  };
>                  return stream(new
>         StreamSource.ForIterator<>(__iterator), StreamOpFlag.IS_ORDERED);
>              }
>
>         Not too difficult.  But, the idea is to make things that are
>         easy in the header of a for-loop to be easy as the source of a
>         stream.
>
>             repeat(n) in Scheme is about 10 characters.
>
>
>         Yeah, well this is Java...
>
>             How difficult is it to implement a merge, as might be needed
>             to generate
>             Hamming numbers? (One of my favorite test cases.)
>
>
>         You mean, interleave two streams?  That's on our list to
>         implement as Streams.interleave(a, b).
>
>             Is there a method to limit a stream to a length?  If so then
>             one of your
>             methods may be extra baggage.
>
>
>         Yes: stream.limit(n).
>
>


More information about the lambda-libs-spec-observers mailing list