Function type naming conventions

Remi Forax forax at univ-mlv.fr
Wed Jan 23 12:34:54 PST 2013


On 01/23/2013 09:19 PM, Sam Pullara wrote:
> Awesome, this is where I hoped we would end up on this.
>
> Thanks,
> Sam

I agree with Sam :)
but I still think that

 > Plus some derived convenience types:
 >
 >   UnaryOperator<T> extends Function<T,T>
 >   BinaryOperator<T> extends BiFunction<T,T,T>

UnaryOperator should be Operator and BinaryOperator should be BiOperator,
it's just more regular.

Operator<T> extends Function<T,T>
BiOperator<T> extends BiFunction<T,T,T>

Rémi

>
> On Jan 23, 2013, at 12:11 PM, Brian Goetz <brian.goetz at oracle.com 
> <mailto:brian.goetz at oracle.com>> wrote:
>
>> Returning to this....
>>
>> Recall that the primary goal in choosing SAM names is user 
>> ergonomics.  Not only do we want to make things clear, but we want to 
>> assign the simple names to the most commonly used configurations. 
>> Hence "Function<T,U>" instead of "ObjectToObjectFunction".  A 
>> secondary goal is for the names to be mechanically derivable from a 
>> reasonably small set of rules.
>>
>> To recap where we are:
>>
>> Base types, each with their own natural arity:
>>
>> Block<T>
>> 	T -> void
>> Function<T,R> 	T -> R
>> Predicate<T> 	T -> boolean
>> Supplier<T> 	() -> T
>>
>>
>> Plus some derived convenience types:
>>
>>   UnaryOperator<T> extends Function<T,T>
>>   BinaryOperator<T> extends BiFunction<T,T,T>
>>
>> Plus arity prefixes to modify the natural arity of these:
>>
>>  BiFunction<T,U,R>     // (T,U) -> R
>>  BiPredicate<T,U>       // (T,U) -> boolean
>>
>> We have a primitive specialization convention that lets you prefix 
>> {Int,Long,Double,Boolean,Obj} on front of one of these (possibly 
>> multiple times).  In accordance with our ergonomics goal, we want 
>> simple specializations for those that specalize return type, since 
>> that has been the most common specialization.  (Obj is used when you 
>> want to specialize later type params.)
>>
>> We first tried ordering the type parameters return-type-first, so 
>> that partial specialization could proceed left-to-right.  This went 
>> over like a lead balloon.  So we then adopted the rule that we 
>> specialize return type first when the return type is generic, and 
>> otherwise left-to-right.  This means that
>>
>>   IntFunction<T>
>>
>> is a function T -> int.  This seemed attractive as T -> int was far, 
>> far more common than int -> T.
>>
>> But this special treatment introduces anomalies.  For example, this 
>> rule would (probably) assign
>>
>>   IntDoubleFunction
>>
>> for double -> int, whereas users will probably expect IntDouble 
>> function to correspond to int -> double based on left-to-right. 
>>  There are ways to further complicate the mapping to avoid this but 
>> that just moves the anomalies into darker corners.
>>
>> Based on offline discussions with Kevin and Joe, we concluded that 
>> backing off slightly from our desire to have the shortest possible 
>> name for the most common specialization can yield a more consistent 
>> naming scheme while still being acceptably ergonomic.
>>
>> So, new proposal:
>>
>>  - When specializing return type, prefix "ToXxx"
>>  - Otherwise specializations gobble type arguments left-to-right.
>>
>> So:
>>
>>   Function<T,U>             // T -> U
>>   DoubleFunction<T>    // double -> T
>>   ToIntFunction<T>        // T -> int
>>   DoubleToIntFunction  // double -> int
>>
>> This also means there is more than one way to represent the same 
>> thing in some cases.  For example, also consistent with these rules are:
>>   DoubleIntFunction    // double -> int
>> since this covers all the type arguments.
>>
>> Only Function and Supplier are generic in return, so specializations 
>> of Block and Predicate would not be affected by the new "To" rule.
>>
>> This scheme is simpler and has fewer anomalies.  The casualty is that 
>> IntFunction becomes ToIntFunction -- arguably clearer, and only 
>> slightly more verbose -- overall seems a win.
>>
>> Here's how the current SAMs would be affected:
>>
>> IntFunction<T>
>> 	ToIntFunction<T>
>> IntBiFunction<T,U>
>> 	ToIntBiFunction<T,U>
>> IntBlock
>> 	not affected
>> IntBinaryOperator
>> 	not affected
>> IntPredicate
>> 	not affected
>> IntSupplier
>> 	ToIntSupplier
>> alternately, not affected
>> IntUnaryOperator
>> 	not affected
>> ObjIntBiBlock
>> 	not affected
>> ObjIntFunction
>> 	IntToObjFunction
>> alternately, IntObjFunction
>>
>>
>> A possible (orthogonal) tweak to this system would be:
>>  - When there are enough specializations to cover all arguments, omit 
>> the arity prefix
>>
>> This would turn ObjIntBiBlock into ObjIntBlock.  Some more 
>> investigation would be required to ensure there are no collisions here.
>



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