- TypeScript Tutorial
- TypeScript Home
- TypeScript Introduction
- TypeScript Setup
- TypeScript First Program
- TypeScript vs JavaScript
- TypeScript Data Types
- TypeScript Type Inference
- TypeScript Type Annotations
- TypeScript Interfaces
- TypeScript Enums
- TypeScript Type Aliases
- TypeScript Type Assertions
- TypeScript Variables
- TypeScript Functions
- TypeScript Functions
- TypeScript Optional Parameters
- TypeScript Default Parameters
- TypeScript Rest Parameters
- TypeScript Arrow Functions
- Classes and Objects
- Introduction to Classes
- Properties and Methods
- Access Modifiers
- Static Members
- Inheritance
- Abstract Classes
- Interfaces vs Classes
- Advanced Types
- TypeScript Union Types
- TypeScript Intersection Types
- TypeScript Literal Types
- TypeScript Nullable Types
- TypeScript Type Guards
- TypeScript Discriminated Unions
- TypeScript Index Signatures
- TypeScript Generics
- Introduction to Generics
- TypeScript Generic Functions
- TypeScript Generic Classes
- TypeScript Generic Constraints
- TypeScript Modules
- Introduction to Modules
- TypeScript Import and Export
- TypeScript Default Exports
- TypeScript Namespace
- Decorators
- Introduction to Decorators
- TypeScript Class Decorators
- TypeScript Method Decorators
- TypeScript Property Decorators
- TypeScript Parameter Decorators
- Configuration
- TypeScript tsconfig.json File
- TypeScript Compiler Options
- TypeScript Strict Mode
- TypeScript Watch Mode
TypeScript Type Inference
TypeScript has a powerful feature called type inference, which automatically determines the type of a variable based on its value. This reduces the need to explicitly specify types while still maintaining type safety.
How Type Inference Works
TypeScript analyzes the context and the value assigned to a variable or function and infers its type. This is particularly helpful for making code cleaner and more concise.
Variable Type Inference
TypeScript infers the type of a variable based on its initial value.
let message = "Hello, TypeScript!";
// TypeScript infers 'message' as a string
console.log(typeof message); // Output: string
If you assign a number to a variable, TypeScript infers it as a number
.
let age = 25;
// TypeScript infers 'age' as a number
console.log(typeof age); // Output: number
Function Return Type Inference
TypeScript automatically infers the return type of a function based on its return
value.
function add(a: number, b: number) {
return a + b;
}
// TypeScript infers the return type as 'number'
let result = add(10, 20);
console.log(result); // Output: 30
If no value is returned, TypeScript infers the return type as void
.
function logMessage(message: string) {
console.log(message);
}
// TypeScript infers the return type as 'void'
logMessage("TypeScript is awesome!");
Contextual Typing
TypeScript can infer types based on the context in which a value is used, such as event handlers or callbacks.
const numbers = [1, 2, 3, 4];
numbers.forEach((num) => {
console.log(num.toFixed()); // TypeScript knows 'num' is a number
});
Here, num
is inferred as a number
because it comes from an array of numbers.
Type Inference in Arrays
TypeScript infers the type of an array based on its elements.
let names = ["Alice", "Bob", "Charlie"];
// TypeScript infers 'names' as an array of strings (string[])
names.push("David"); // Valid
names.push(123); // Error: Argument of type 'number' is not assignable to type 'string'
Type Inference with const
When a variable is declared with const
, TypeScript infers the most specific type, also known as a literal type.
const color = "blue";
// TypeScript infers 'color' as the literal type 'blue'
console.log(color); // Output: blue
Best Common Type
When inferring the type of a variable with mixed types, TypeScript determines the best common type.
let mixed = [1, "two", true];
// TypeScript infers 'mixed' as (string | number | boolean)[]
console.log(mixed);
Summary
TypeScript's type inference system automatically determines types based on initial values, function returns, and contextual usage. This allows developers to write cleaner, more readable code while still benefiting from strong type checking. Explicit type annotations are not always necessary, but they can be added when needed for clarity.