TypeScript Unions
last modified February 24, 2025
Unions in TypeScript allow a variable to be one of several types. They are
declared using the |
operator. Unions provide flexibility and
power in type checking, making it possible to write more robust code.
Unions in TypeScript are a way to define a variable that can be one of several types. For example, a variable can be either a string or a number.
Declaring Unions
This example demonstrates how to declare a union in TypeScript.
let value: string | number = "Hello"; value = 10; console.log(value); // Output: 10
The value
variable is declared to be of type string
or
number
. It is initially assigned the string value "Hello
",
but can later be assigned the number value 10
.
Union Narrowing
TypeScript uses a process called union narrowing to determine the actual type of a union variable.
function printValue(value: string | number) { if (typeof value === "string") { console.log(value.length); } else { console.log(value); } } printValue("Hello"); // Output: 5 printValue(10); // Output: 10
In this example, the printValue
function takes a union argument.
By checking the type of value
using typeof
, TypeScript
narrows down the possible types, allowing us to call methods specific to that
type.
Union Operators
TypeScript provides several operators for working with unions.
Intersection
The intersection operator (∧) combines multiple types into a single type.
interface Person { name: string; age: number; } interface Developer { language: string; } type Programmer = Person & Developer; let programmer: Programmer = { name: "Jan", age: 35, language: "TypeScript" };
Distributive Conditional Types
Distributive conditional types apply a type operation to each member of a union.
Distributive union types apply a conditional type to each member of a union type individually, and then combine the results back into a union type. This is achieved through conditional types using the extends keyword.
type MyUnion = number | boolean | string; type ToArray<T> = T extends any ? T[] : never; type MyUnionArray = ToArray<MyUnion>; const numberArray: MyUnionArray = [1, 2, 3]; const booleanArray: MyUnionArray = [true, false]; const stringArray: MyUnionArray = ["a", "b", "c"]; console.log(numberArray); console.log(booleanArray); console.log(stringArray);
In this example, the ToArray
conditional type transforms each
member of the MyUnion union type into an array of that member type. The
resulting MyUnionArray type will be number[] | boolean[] | string[]
,
meaning it can be an array of numbers, booleans, or strings.
Mapped Types
Mapped types apply a type transformation to each property of an object type.
type ReadOnly<T> = { readonly [P in keyof T]: T[P]; }; interface Person { name: string; age: number; } type ReadOnlyPerson = ReadOnly<Person>; let person: ReadOnlyPerson = { name: "Jan", age: 35 }; person.name = "John"; // Error: Cannot assign to 'name' because it is a read-only property.
Best Practices for Using Unions
- Use Union Narrowing: Utilize TypeScript's union narrowing feature to determine the actual type of a union variable.
- Prefer Specific Types: Avoid unions when possible, as they can make code harder to understand and maintain.
- Use Union Operators: Leverage TypeScript's union operators, such as intersection and mapped types, for powerful type manipulation.
- Handle Edge Cases: Always check for edge cases when working with unions, as they can introduce unexpected behavior.
Source
TypeScript Unions and Intersections Documentation
In this article, we have explored TypeScript unions and demonstrated their usage through practical examples.
Author
List all TypeScript tutorials.