Bikeshed opportunity: compose vs composeWith

Remi Forax forax at univ-mlv.fr
Tue Nov 27 07:16:56 PST 2012


On 11/26/2012 11:57 PM, Brian Goetz wrote:
>>> I like the "then" convention to indicate sequencing.  In context:
>>>
>>> people.sort(comparing(Person::getLast)
>>>               .thenCompare(comparing(Person::getFirst)))
>>
>> or
>>    people.sort(comparing(Person::getLast, Person::getFirst))
>> (comparing is a static method so it can be annotated with @SafeVarargs).
>
> This one falls apart as there is no common supertype between
>   Function<Person, T extends Comparable>
> and
>   IntFunction<Person>
>
> so people will try
>
>   people.sort(comparing(Person::getLast, Person::getHeight))
>
> and be mystified by the error message they get.  And then ask for a 
> combinatorial explosion of comparing() methods.

Ok, trying to implement it ...

public class PrimitiveFunctions {
   interface Function<T, U> {
     public U apply(T element);
   }

   interface IntFunction<T> extends Function<T, Integer> {
     @Override
     public default Integer apply(T element) {
       return applyInt(element);
     }

     public int applyInt(T element);
   }

   @SafeVarargs
   @SuppressWarnings("unchecked")
   public static <T> Comparator<T> comparing(Function<? super T, ? 
extends Comparable<?>>... functions) {
     if (functions.length == 0) {
       throw new IllegalArgumentException();
     }
     return (o1, o2) -> {
       for(Function<? super T, ? extends Comparable<?>> function: 
functions) {
         int diff = 
((Comparable)function.apply(o1)).compareTo(function.apply(o2));
         if ( diff != 0) {
           return diff;
         }
       }
       return 0;
     };
   }

   public static void main(String[] args) {
     Comparator<String> c1 = comparing(s -> s.length(), s -> s.charAt(0));
     Comparator<String> c2 = comparing((String s) -> s.length(), (String 
s) -> s.charAt(0));
     Comparator<String> c3 = comparing(String::length, String::hashCode);
     //System.out.println(c.compare("foo", "bar"));
   }
}

c1 doesn't compile, comparing is a varargs and the lambda doesn't define 
its parameter type,
so there is a cyclic inference here. This is a known issue of current 
inference algorithm
(see an old discussion with Dan about that). I think it's fixable, but 
I'm not sure

c2 compiles :)

c3 doesn't and I don't understand why.
PrimitiveFunctions.java:38: error: method comparing in class 
PrimitiveFunctions cannot be applied to given types;
     Comparator<String> c3 = comparing(String::length, String::hashCode);
                             ^
   required: Function<? super T,? extends Comparable<?>>[]
   found: String#length,String#hashCode
   reason: Cannot instantiate inference variables T because of an 
inference loop
   where T is a type-variable:
     T extends Object declared in method <T>comparing(Function<? super 
T,? extends Comparable<?>>...)
1 error

cheers,
Rémi





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