Collector / Collectors

Brian Goetz brian.goetz at oracle.com
Fri Aug 9 12:47:16 PDT 2013


Having spent a little more time with this and not found a better 
solution, I think the answer is this: collectingAndThen(coll, fn).

Proposed spec (implementation is trivial):

     /**
      * Adapts a {@code Collector} to perform an additional finishing
      * transformation.  For example, one could adapt the {@link #toList()}
      * collector to always produce an immutable list with:
      * <pre>{@code
      *     List<String> people
      *         = people.stream().collect(collectingAndThen(toList(), 
Collections::unmodifiableList));
      * }</pre>
      *
      * @param <T> the type of the input elements
      * @param <A> intermediate accumulation type of the downstream 
collector
      * @param <R> result type of the downstream collector
      * @param <RR> result type of the resulting collector
      * @param finisher a function to be applied to the final result of 
the downstream collector
      * @param downstream a collector
      * @return a collector which performs the action of the downstream 
collector,
      * followed by an additional finishing step
      */
     public static<T,A,R,RR> Collector<T,A,RR> 
collectingAndThen(Collector<T,A,R> downstream,
 
Function<R,RR> finisher) {



On 7/9/2013 1:32 AM, Brian Goetz wrote:
> Returning to this...
>
> We tried the collector.andThen() approach.  People liked the API, but we
> ran afoul of type inference issues.  I think the remaining option is a
> combinator:
>
>    collector+finisher -> collector
>
> but the previously suggested name (finishing) had an "inside out" feel.
>   But, I think a slight name tweak will do the job:
>
>       groupingBy(Person::getCity,
>                  collectingAndThen(maxBy(comparing(Person::getHeight)),
>                                    Optional::get)));
>
> to suggest what is going on.
>
> On 6/26/2013 2:26 PM, Brian Goetz wrote:
>>
>> 2.  Making the one-arg reducing(), and minBy/maxBy, return Optional
>> means that queries like "tallest person by city" end up with Optional in
>> the value:
>>
>>    Map<City, Optional<Person>> m
>>      = people.collect(groupingBy(Person::getCity,
>>                                  maxBy(comparing(Person::getHeight)));
>>
>> Which is doubly bad because the optionals here will *always* be present,
>> since otherwise there'd be no associated key.
>>
>> I can see a few ways to address this:
>>   - Provide non-optional versions of minBy/maxBy/reducing, which would
>> probably have to return null for "no elements".
>>   - Provide a way to add a finisher to a Collector, which is more
>> general (and could be used, say, to turn toList() into something that
>> always collects to an immutable list.)
>>
>> The latter could, in turn, be done in one of two ways:
>>
>>   - Add a andThen(f) method to Collector.  Then the above would read:
>>
>>     groupingBy(Person::getCity,
>>                maxBy(comparing(Person::getHeight))
>>                  .andThen(Optional::get))
>>
>>   - Add a combinator:
>>
>>     groupingBy(Person::getCity,
>>                finishing(maxBy(comparing(Person::getHeight)),
>>                          Optional::get)));
>>
>> I prefer the former because it reads better in usage; using a combinator
>> function feels a little "inside out."


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