Performance model of sorted() ?

Paul Sandoz paul.sandoz at oracle.com
Mon Nov 4 01:13:08 PST 2013


On Nov 1, 2013, at 7:18 PM, Brian Goetz <brian.goetz at Oracle.COM> wrote:

> The "slicing" part is right, but the single-threaded assumption is wrong.
> 
> What happens is that the upstream stateless ops will be done in 
> parallel, with the results being collected at the leaves of the 
> computation tree, into a bunch of mini-buffers.  Now, each will be 
> sorted (in parallel), and then we begin a parallel merge, collecting the 
> results into an array.  Then the downstream stateless ops and the 
> terminal op are run, in parallel.
> 

That is what we wanted/planned to do :-) but for reasons of expediency we currently resort to collecting, in parallel, to an array and then using Arrays.parallelSort e.g. the equivalent of:

  Stream s = ...
  s = s.parallel();
  Object[] a = s.toArray();
  Arrays.parallelSort(a);
  s = Arrays.stream(a);

So there is some redundancy and Arrays.parallelSort will not go parallel if the size of the array is less than 2^13.

Unfortunately it will be tricky to share any code from Arrays.parallelSort, thus requiring duplication x 4 (for refs + primitives). (Arrays.parallelArray splits into quarters to ensure the final sort is in the main array but in our case i don't think we require that.)

For an optimal implementation that computation tree needs to consist of nodes that are F/J tasks (such that collection and merging can happen concurrently) and we need to support sized and non-sized pipelines slightly differently for the collection phase. Given that we can use Arrays.parallelArray code as a template i think most of the hard work is done.

Paul.

> So, no truly sequential steps imposed by the framework (though binary 
> parallel merges are generally dominated by the last merge step.)



More information about the lambda-dev mailing list