Forms for reduce() -- part 1
Brian Goetz
brian.goetz at oracle.com
Fri Dec 14 12:15:16 PST 2012
>> Open question: do we keep the name mutableReduce to make it clear what
>> is going on?
>
> If you went without the name, I'm not sure how clear it will be with the existing types that one is mutating or non mutating. Would this make sense, or not so much?
>
> <U> U reduce(U zero,
> BiFunction<U, T, U> nonMutatingAccumulator,
> BinaryOperator<U> nonMutatingReducer);
>
> <R> R reduce(Supplier<R> seedFactory,
> BiBlock<R, T> mutatingAccumulator,
> BiBlock<R, R> mutatingReducer);
There's no reason this doesn't *work*. My concern was that calls of
both will involve lambdas of two arguments, and it will be less obvious
by reading the code whether (a, b) -> goo is being used as a functional
reducer or a mutable accumulator. The parameter names will not be
present at the use site, so readers of the code will have to reason
about whether this is a functional reduce or a mutable inject.
So, the question remains -- is the word "mutable" (or some other way of
saying that) a helpful guide about what is being done here, or pedantic
noise that will irritate the users? (Note that I think we should care
much less how it makes people feel when *writing* code than how it helps
comprehension when *reading* code.)
> These are APIs we've had in a separate library for several years that we've moved into GS Collections RichIterable interface and which will become available in our 3.0 release. They are a combination of groupBy/injectInto so certainly have some differences to reduce, but they also have some similarities in terms of supporting mutating/non-mutating versions. We chose not to go with the name mutableAggregateBy, but instead named the parameter mutatingAggregator or nonMutatingAggregator. We felt the difference in type made it pretty clear (Function2 vs. Procedure2).
The type does make it clear when the call site has "new Function() { ...
}". Which of course described many call sites in a pre-lambda world.
In a post-lambda world, the type is inferred, and both functions and
blocks often look similar at the use site:
(a, b) -> a+b
(a, b) -> a.add(b) // where add(b) is void-bearing and mutative
More information about the lambda-libs-spec-observers
mailing list