-
Notifications
You must be signed in to change notification settings - Fork 1.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Question: What is the purpose of this cast? #42889
Comments
The cast does not appear if the same function is written as a regular class method and you tear it off. class GenericType<T> {
Null genericFunction(T t) => null;
}
main() {
var generic = GenericType<int>();
var tearOff = generic.genericFunction;
}
|
In the example there is an implicit cast because class GenericType<T> {
var genericFunction = (T t) => null; // Inferred type `Null Function(T)`.
}
main() {
GenericType<num> generic = GenericType<int>();
var tearOff = generic.genericFunction; // Subject to a dynamic type check.
void Function(num) f = tearOff; // Same static type.
f(3.15);
} I've changed the example to have an explicitly declared type on the local variable Dart uses dynamically checked covariance, so However, There are several ways to handle this. The approach taken in Dart is to insert a caller side type check at the point where The real solution is to introduce sound variance (declaration site, use site), and using that feature to ensure that The situation is similar but not identical if we change the example such that class GenericType<T> {
Null genericFunction(T t) => null;
}
main() {
GenericType<num> generic = GenericType<int>();
var tearOff = generic.genericFunction;
} In this case the trade-off needed because of dynamically checked covariance is handled by giving the torn-off method a different type: So the function object obtained by tearing off a method whose signature has a type parameter in a contravariant position will have a type that claims that it will accept any object as an argument. The static type of The approaches used for the two cases (1: returning function objects whose type is contravariant in a type parameter, and 2: tearing off methods whose signature has a type parameter in a contravariant position) cannot be used for the "opposite" case: 1: It would be a massively breaking change to perform a dynamic type check during a tear-off operation for every method with such a signature (and require it to have the naive static type where covariance is ignored), and given that regular invocations of the method will perform the same dynamic type check as the torn off function object, the amount of dynamic checking is unchanged. 2: An instance variable can hold any function object, so in the case where that instance variable has a type which is contravariant in a type parameter it would require a wrapper function if we were to treat it the same as methods (the wrapper would give the function object the "erased" run-time type that torn-off methods get, and it would perform the dynamic type checks on actual arguments). Creating and invoking wrappers is costly in terms of performance, and it creates delicate problems with object identity and equality. Again, we can eliminate the erasure on the dynamic type of torn-off methods when the relevant type parameters have been annotated with variance markers (with some extra constraints, e.g., the method can't be inherited from a superinterface using dynamically checked covariance), so sound variance is again the real solution. |
Thanks for the detailed explanation! I think I keep reforming the same question in different ways and the answer is consistently that dart needs sound variance for generic type arguments to avoid it. I appreciated your presentation on use site variance when you were last in Seattle. Hopefully we can find a way to fit it into the language. I'm closing this issue since the check is clearly needed still. |
I noticed simply tearing off a field that contains a function from a generic class produces an
AsExpression
node when the function uses the generic type argument in the contravariant position. The evaluation ofTypeEnvironment.isSubtypeOf(from, to, SubtypeCheckMode.withNullabilities)
evaluates totrue
but since the cast is marked with theTypeError
andCovarianceCheck
flags ddc will emit a cast that always seems to pass.Is this cast necessary and is there an example that would cause this cast to fail?
@johnniwinther @eernstg
The text was updated successfully, but these errors were encountered: