Comparator combinators
Sam Pullara
spullara at gmail.com
Wed Aug 21 13:24:27 PDT 2013
Yes, I'd like to have it in the method name.
Sam
On Aug 21, 2013, at 12:13 PM, Brian Goetz <brian.goetz at oracle.com> wrote:
> To clarify, you would rather disambiguate with the method name than have to provide additional type information (explicit lambda, type witnesses, or cast), right?
>
> On 8/21/2013 3:00 PM, Sam Pullara wrote:
>> I agree. I would much rather end up accidentally boxing than having to
>> disambiguate everywhere.
>>
>> Sam
>>
>> On Aug 21, 2013, at 11:33 AM, Tim Peierls <tim at peierls.net
>> <mailto:tim at peierls.net>> wrote:
>>
>>> Explicit comparingXxx, etc. is best.
>>>
>>>
>>> On Wed, Aug 21, 2013 at 2:28 PM, Brian Goetz <brian.goetz at oracle.com
>>> <mailto:brian.goetz at oracle.com>> wrote:
>>>
>>> Waiting for input on this.
>>>
>>> I'm torn. On the one hand, we'd be publishing a new,
>>> lambda-centric API that doesn't conform to our "safe overloading"
>>> rules, and for which users would have to use explicit lambdas in
>>> the cases where method references aren't practical.
>>>
>>> On the other, if we rename the primitive versions to
>>> comparingXxx/thenComparingXxx, we're creating a different sort of
>>> risk -- users will say
>>>
>>> comparing(Person::getAge)
>>>
>>> and get boxing when they didn't intend (this is
>>> comparing(Function<Person,__Integer>), when they probably wanted
>>> comparingInt(ToIntFunction<__Person>)). Though this would be easy
>>> for IDEs to detect and suggest a replacement.
>>>
>>> I think the responsible thing to do is still probably to
>>> disambiguate by method name -- comparingInt, thenComparingLong,
>>> etc. Which isn't pretty but seems to be the new equilibrium
>>> (overloading and inference are in opposition; adding more
>>> inference means we can tolerate less overloading.)
>>>
>>> If there's no input, an executive decision will be made...
>>>
>>>
>>>
>>>
>>> On 8/14/2013 3:49 PM, Brian Goetz wrote:
>>>
>>> This may well be our last API loose end...
>>>
>>> We currently have a pile of Comparator combinators, all
>>> currently called
>>> comparing() or thenComparing(). Regardless of whether we
>>> choose to go
>>> forward with the simplified overloading plan, these overloads
>>> have a
>>> problem: for implicit lambdas, we can't distinguish between
>>>
>>> comparing(T->U)
>>> and
>>> comparing(T->int)
>>>
>>> because we don't type-check the body of the lambda until we do
>>> overload
>>> selection, and don't do overload selection based on whether
>>> there are
>>> type errors in the body (this was roundly rejected as being too
>>> brittle). So for lambdas like:
>>>
>>> comparing(x -> x.size())
>>>
>>> we've got a circularity -- even under the older, more complex
>>> scheme.
>>>
>>> We'd thought perhaps that, if we adopted the heuristic that we can
>>> eagerly give a type to method reference that refers to a
>>> non-overloaded
>>> method (e.g., Person::getAge), then cases like
>>>
>>> comparing(Person::getAge)
>>>
>>> can be handled, and this might take some of the pressure off
>>> the problem.
>>>
>>> For lambdas (with either approach) you can always get what you
>>> want with
>>> an explicit lambda:
>>>
>>> comparing((Person p) -> p.getAge())
>>>
>>> since this can be type-checked early.
>>>
>>> So the question is, is this good enough, even though it falls
>>> afoul of
>>> the overloading guidelines for implicit lambdas? Or, should
>>> we mangle
>>> the names of the methods?
>>>
>>> This question comes down to whether we think its better to force
>>> everyone to explicitly pick a method, but then supporting
>>> implicit lambdas:
>>>
>>> comparingInt(x -> x.size())
>>>
>>> or forcing users to use non-ambigous method refs or explicit
>>> lambdas:
>>>
>>> comparing(Person::getAge)
>>> or
>>> comparing((Person p) -> p.getAge())
>>>
>>> Which disambiguation approach is worse?
>>>
>>>
>>
More information about the lambda-libs-spec-experts
mailing list