- 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
Introduction to TypeScript Generics
Generics in TypeScript provide a powerful way to write reusable and type-safe code. They allow you to define functions, classes, and interfaces that can work with any data type while still enforcing type safety. By using generics, you can make your code more flexible and maintainable without sacrificing type safety.
What are Generics?
Generics are a feature in TypeScript that allows you to create components (such as functions, classes, or interfaces) that can work with multiple data types. Instead of specifying a concrete data type like string
or number
, you define a placeholder type that can be replaced with any type when the component is used.
Generics are denoted by angle brackets (< >
) and typically involve a type parameter (usually represented as T
or other identifiers).
Why Use Generics?
- Reusability: Generics allow you to write functions, classes, or interfaces that can operate on various types without duplicating code.
- Type Safety: Generics maintain the benefits of static typing while keeping your code flexible, ensuring that the types are checked at compile time.
- Avoiding Code Duplication: With generics, you can write functions or classes that work with different types, eliminating the need to write multiple versions of the same function or class for different data types.
Syntax of Generics
A generic is defined by adding a type parameter inside angle brackets (<T>
), where T
is a placeholder for any type.
function identity<T>(value: T): T {
return value;
}
<T>
: This is the type parameter.T
can be replaced by any type when the function is called.value: T
: The input parameter's type is defined by the type parameterT
.: T
: The return type of the function is also of typeT
.
Example of Generics
1. Generic Function
Here is a simple example of a generic function that returns the same value passed into it:
function identity<T>(value: T): T {
return value;
}
console.log(identity(42)); // Output: 42
console.log(identity("Hello")); // Output: Hello
- In this example, the function
identity
accepts a value of any type (T
) and returns a value of the same type. - When you call
identity(42)
, TypeScript infers thatT
isnumber
. - When you call
identity("Hello")
,T
is inferred asstring
.
2. Generic with Multiple Type Parameters
You can also define functions that work with multiple types. For instance, if you want to create a function that swaps two values:
function swap<T, U>(a: T, b: U): [U, T] {
return [b, a];
}
const swapped = swap(1, "one");
console.log(swapped); // Output: ["one", 1]
- This function takes two arguments,
a
andb
, which are of typesT
andU
, respectively. - It returns a tuple where the first element is of type
U
and the second element is of typeT
.
Generic Classes
You can also use generics in classes. Here’s an example of a class that stores a value and has a method to return that value:
class Box<T> {
private value: T;
constructor(value: T) {
this.value = value;
}
getValue(): T {
return this.value;
}
}
const numberBox = new Box(42);
console.log(numberBox.getValue()); // Output: 42
const stringBox = new Box("Hello");
console.log(stringBox.getValue()); // Output: Hello
- The
Box
class is a generic class that accepts a type parameterT
. - The constructor and the
getValue
method use this type parameter to define the type of thevalue
property.
Generic Interfaces
Generics can also be used in interfaces to define flexible yet type-safe structures.
interface Pair<T, U> {
first: T;
second: U;
}
const pair: Pair<number, string> = { first: 1, second: "one" };
console.log(pair.first); // Output: 1
console.log(pair.second); // Output: one
- The
Pair
interface uses two type parameters,T
andU
, to define a pair of values with different types. - When creating the
pair
object, you specify thatT
isnumber
andU
isstring
.
Constraints on Generics
Sometimes, you may want to restrict the types that can be used with a generic. You can do this by using constraints. This ensures that the type passed into the generic is compatible with a specific type or interface.
function loggingIdentity<T extends { length: number }>(value: T): T {
console.log(value.length); // Works because T is guaranteed to have a length property
return value;
}
loggingIdentity("Hello"); // Output: 5
loggingIdentity([1, 2, 3]); // Output: 3
- In this example, the type parameter
T
is constrained to types that have alength
property (such asstring
orarray
). - This ensures that the
length
property is available onvalue
.
Generic Defaults
You can provide a default type for a generic, which will be used if no type is specified when the function or class is called.
function wrapInArray<T = string>(value: T): T[] {
return [value];
}
console.log(wrapInArray(5)); // Output: [5]
console.log(wrapInArray("Hello")); // Output: ["Hello"]
- In this case, the default type for
T
isstring
, so if no type is provided, TypeScript will assumeT
isstring
.
Summary
Generics in TypeScript are a powerful feature that allows you to write reusable, flexible, and type-safe code. By using generics, you can create functions, classes, and interfaces that work with any data type while still maintaining strong typing. Generics enhance code reusability, provide better type safety, and make your code more maintainable.
- Generic Function: A function that can work with different types while maintaining type safety.
- Generic Classes and Interfaces: Classes and interfaces that are flexible enough to handle different types.
- Constraints: Restrict the types that can be used with a generic, making your code more predictable.
- Default Types: Provide default types for generics, allowing for simpler syntax while still being type-safe.
Generics are a key part of making your TypeScript code more scalable and reusable while avoiding code duplication.