Type Inference

#code/DYUwLgBAHhC8EGYDcAoA9GiEB6B+IA)

let x = 3;

let x: number

The type of the x variable is inferred to be number. This kind of inference takes place when initializing variables and members, setting parameter default values, and determining function return types.

In most cases, type inference is straightforward. In the following sections, we’ll explore some of the nuances in how types are inferred.

Best common type

When a type inference is made from several expressions, the types of those expressions are used to calculate a “best common type”. For example,

Try this code ↗

let x = [0, 1, null];

let x: (number | null)[]

To infer the type of x in the example above, we must consider the type of each array element. Here we are given two choices for the type of the array: number and null. The best common type algorithm considers each candidate type, and picks the type that is compatible with all the other candidates.

Because the best common type has to be chosen from the provided candidate types, there are some cases where types share a common structure, but no one type is the super type of all candidate types. For example:

Try this code ↗

let zoo = [new Rhino(), new Elephant(), new Snake()];

let zoo: (Rhino | Elephant | Snake)[]

Ideally, we may want zoo to be inferred as an Animal[], but because there is no object that is strictly of type Animal in the array, we make no inference about the array element type. To correct this, explicitly provide the type when no one type is a super type of all other candidates:

Try this code ↗

let zoo: Animal[] = [new Rhino(), new Elephant(), new Snake()];

let zoo: Animal[]

When no best common type is found, the resulting inference is the union array type, (Rhino | Elephant | Snake)[].

Contextual Typing

Type inference also works in “the other direction” in some cases in TypeScript. This is known as “contextual typing”. Contextual typing occurs when the type of an expression is implied by its location. For example:

Try this code ↗

window.onmousedown = function (mouseEvent) {
console.log(mouseEvent.button);
console.log(mouseEvent.kangaroo);
};
Generated error
Property 'kangaroo' does not exist on type 'MouseEvent'.

Here, the TypeScript type checker used the type of the Window.onmousedown function to infer the type of the function expression on the right hand side of the assignment. When it did so, it was able to infer the type ↗ of the mouseEvent parameter, which does contain a button property, but not a kangaroo property.

This works because window already has onmousedown declared in its type:

// Declares there is a global variable called 'window'
declarevarwindow: Window & typeofglobalThis;

// Which is declared as (simplified):
interfaceWindowextendsGlobalEventHandlers {
// ...
}

// Which defines a lot of known handler events
interfaceGlobalEventHandlers {
onmousedown: ((this: GlobalEventHandlers, ev: MouseEvent) =>any) | null;
// ...
}

TypeScript is smart enough to infer types in other contexts as well:

Try this code ↗

window.onscroll = function (uiEvent) {
console.log(uiEvent.button);
};
Generated error
Property 'button' does not exist on type 'Event'.

Based on the fact that the above function is being assigned to Window.onscroll, TypeScript knows that uiEvent is a UIEvent ↗, and not a MouseEvent ↗ like the previous example. UIEvent objects contain no button property, and so TypeScript will throw an error.

If this function were not in a contextually typed position, the function’s argument would implicitly have type any, and no error would be issued (unless you are using the noImplicitAny option):

Try this code ↗

const handler = function (uiEvent) {
console.log(uiEvent.button); // <- OK
};

We can also explicitly give type information to the function’s argument to override any contextual type:

Try this code ↗

window.onscroll = function (uiEvent: any) {
console.log(uiEvent.button); // <- Now, no error is given
};

However, this code will log undefined, since uiEvent has no property called button.

Contextual typing applies in many cases. Common cases include arguments to function calls, right hand sides of assignments, type assertions, members of object and array literals, and return statements. The contextual type also acts as a candidate type in best common type. For example:

Try this code ↗

function createZoo(): Animal[] {
return [newRhino(), newElephant(), newSnake()];
}

In this example, best common type has a set of four candidates: Animal, Rhino, Elephant, and Snake. Of these, Animal can be chosen by the best common type algorithm.

Last updated on