Advanced Type Features in TypeScript

November 17th, 2022 | ☕️ 5 min read
Photo by Lerone Pieters, powered by unsplash.

TypeScript Fundamental Series

  1. Understanding Types in TypeScript
  2. Mastering Functions in TypeScript
  3. Exploring Interfaces in TypeScript
  4. Utilizing Classes in TypeScript
  5. Leveraging Generics in TypeScript
  6. Advanced Type Features in TypeScript

Welcome to the sixth and final post in our series on the fundamentals of TypeScript. In this article, we'll explore some advanced type features that can further enhance your TypeScript development experience.

While the previous blog posts covered the core concepts of types, functions, interfaces, classes, and generics, there are a few more powerful type features that are worth learning about. Let's dive in!

Union and Intersection Types TypeScript provides two powerful type features: union types and intersection types. These allow you to create more complex and expressive types.

Union types represent a value that can be one of several types:

let age: number | string = 30; // age can be a number or a string age = '40'; // this is also valid

In this example, the age variable can hold either a number or a string value.

Intersection types combine multiple types into a single type:

interface Admin { id: string; admin: boolean; } interface User { id: string; name: string; } type AdminUser = Admin & User; let adminUser: AdminUser = { id: 'abc123', admin: true, name: 'John Doe' };

In this example, the AdminUser type is an intersection of the Admin and User interfaces, meaning it must have all the properties of both interfaces.

Type Aliases TypeScript allows you to create your own custom types using type aliases. This can be particularly useful when you want to give a meaningful name to a complex type.

type ID = string; type Person = { id: ID; name: string; age: number; }; let john: Person = { id: 'abc123', name: 'John Doe', age: 30 };

In this example, we've created a Person type alias that represents an object with id, name, and age properties. We've also created a ID type alias for the id property, which is a string.

Literal Types TypeScript also supports literal types, which allow you to specify exact values that a variable or parameter can hold. This can be particularly useful when working with APIs or configuration settings.

type Color = 'red' | 'green' | 'blue'; let myColor: Color = 'red'; // Valid // myColor = 'yellow'; // Error: Type '"yellow"' is not assignable to type 'Color'.

In this example, the Color type is a union of three literal string types: 'red', 'green', and 'blue'. This ensures that the myColor variable can only be assigned one of those three values.

Conditional Types TypeScript's conditional types allow you to create types that depend on other types. This can be a powerful tool for creating more dynamic and expressive type systems.

type Flatten<T> = T extends Array<infer U> ? U : T; type FlattenedArray = Flatten<number[]>; // FlattenedArray is number type FlattenedValue = Flatten<number>; // FlattenedValue is number

In this example, the Flatten type is a conditional type that checks if the input type T is an array. If it is, the output type is the element type of the array (U). If it's not an array, the output type is simply T.

Mapped Types Mapped types in TypeScript allow you to create new types by transforming the properties of an existing type. This can be useful for creating variations of existing types or for building generic utility types.

interface Person { name: string; age: number; } type ReadonlyPerson = Readonly<Person>; // ReadonlyPerson is { readonly name: string; readonly age: number; } type PartialPerson = Partial<Person>; // PartialPerson is { name?: string; age?: number; }

In this example, the Readonly and Partial utility types are mapped types that create new types based on the Person interface.

Utility Types TypeScript provides a set of built-in utility types that can help you work with more complex type scenarios. Some of the most useful utility types include:

  • Pick<T, K>: Creates a type by picking a set of properties K from an existing type T.
  • Omit<T, K>: Creates a type by omitting a set of properties K from an existing type T.
  • Exclude<T, U>: Creates a type by excluding from T all types that are assignable to U.
  • NonNullable<T>: Creates a type by excluding null and undefined from T.

These utility types can help you write more concise and expressive code, especially when working with complex data structures.

By exploring these advanced type features in TypeScript, you'll be able to create more robust, maintainable, and flexible code. TypeScript's type system is a powerful tool that can greatly improve the quality and reliability of your applications.

We've covered a lot of ground in this series, from the basics of types to more advanced concepts like generics and conditional types. I hope this has been a valuable journey for you, and that you feel more confident in your TypeScript skills. Happy coding!

Discuss on Twitter