-
Notifications
You must be signed in to change notification settings - Fork 12.8k
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
Typescript type checks passing a callback with non-optional parameter for optional one, but throws error at runtime #18466
Comments
When you include the type annotations, TypeScript evaluates the assignability of the parameter types. function f(callback: (a: string, b?: string) => void) {
callback("a", undefined);
}
const cb = (a: string, b: string) => {
console.log(a.length, b.length);
}
f(cb);
f((a, b) => {
console.log(a.length, b.length);
}); |
I'm new to TypeScript, but I expected that when determining if a function is a substitute for another one, its parameters should be more general (contra-variant on input parameters and co-variant on the output). Why TypeScript does not adhere to this rule? |
TypeScript contravariance and covariance of arguments is likely the biggest issue with TypeScript that doesn't have a clean solution. There are several issues open for this and the conversations is rather extensive and exhaustive. Needless to say, it isn't like other type systems. The FAQ does a decent job of covering off how the type system behaves. This sort of falls under Why are functions with fewer parameters assignable to functions that take more parameters. I will admit I only have semi-good grasp of covariance/contravariance. I am sure some of the core team though can explain it better than I and why it is the way it is. |
Following on from @kitsonk - The part of the FAQ which explains this behaviour is why-are-function-parameters-bivariant |
Thanks for the link to the FAQ. I was a bit confused, I guess structural typing adds another level of complexity. For return values of functions, do you consider them only co-variant. Is it true? |
@afroozeh correct. Return values are treated as co-variant. interface Pet {
feed(): void;
}
interface Cat extends Pet {
purr(): void;
}
declare function getPet(): Pet;
declare function getCat(): Cat;
const p : Pet = getCat(); // type 'Cat' is assignable to type 'Pet'
const gC: () => Pet = getCat; // type '() => Cat' is assignable to type '() => Pet'
const c : Cat = getPet(); // ERROR: Type 'Pet' is not assignable to type 'Cat'
const gp: () => Cat = getPet; // ERROR: Type '() => Pet' is not assignable to type '() => Cat' |
@afroozeh I stand for your point. And @kitsonk , I think the issue here has nothing to with the length of the parameters, but with the type of the parameters. I will show you how the tsc confuses itself when it comes function type assignments. First, let's see when it does the right thing: // the wrapper function ensures tsc doesn't infer the real type of strOrUndefined beforehand.
function wrapper(strOnly: string, strOrUndefined: string | undefined) {
strOnly = strOrUndefined; // error, because undefined is not assignable to string
strOrUndefined = strOnly; // fine, because string is subtype of string | undefined
} But there is a way to trick the compiler failing to detect the error of // the wrapper function ensures tsc doesn't infer the real type of strOrUndefined beforehand.
function wrapper(strOnly: string, strOrUndefined: string | undefined) {
// strOnly = strOrUndefined; // error, because undefined is not assignable to string
const func1 = function (str: string): void {
strOnly = str;// NO ERROR! but str actualy points to strOrUndefined!
}
const func2: (str: string | undefined) => void = func1;
func2(strOrUndefined);// what actually got executed is func1(strOrUndefined);
} Obviously, there is a problem here. And the problem lies on When we assign Why are we doing the opposite by this manner? Because when it comes functions, the real assignment doesn't take place until the functions get executed. Let's see what really happened when func2 got executed. func2 got the parameter of This is absolutely a bug of the compiler, a really misleading one. |
TypeScript Version: 2.4.1
Code
Expected behavior:
Compile error with when compiled with the
strict
flag.(a: string, b: string) => void
is not a subtype of(a: string, b?: string) => void
asstring
is not a supertype ofstring | undefined
, I expected a compile error.Actual behavior:
Compiles, but throws error at runtime
The text was updated successfully, but these errors were encountered: