Optimizing TypeScript Code with Lesser-Known Type Utilities

Banggi Bima Edriantino

March 1, 2025

6 min read

Tidak tersedia dalam Bahasa Indonesia.

Introduction#

TypeScript provides powerful utilities to improve type safety and flexibility in codebases. While commonly used utilities like Partial<T> and Readonly<T> are well known, TypeScript offers lesser-known type utilities that can optimize and streamline development.

This article explores these utilities, how they work, and how they can be applied to write cleaner and more efficient TypeScript code.

1. Extract<T, U> - Picking Specific Types#

The Extract<T, U> utility filters a union type T and keeps only the types that match U.

Example: Extracting Specific Types from a Union#
type Status = "success" | "error" | "pending";
type ErrorStates = Extract<Status, "error" | "pending">;

// Result: "error" | "pending"

Use case: When working with APIs that return multiple states, Extract helps isolate the relevant ones.

2. Exclude<T, U> - Removing Specific Types#

The Exclude<T, U> utility does the opposite of Extract—it removes specific types from a union.

Example: Filtering Out Unwanted Types#
type Status = "success" | "error" | "pending";
type SuccessState = Exclude<Status, "error" | "pending">;

// Result: "success"

Use case: Useful when filtering out error states or specific values from unions.

3. NonNullable<T> - Removing null and undefined#

The NonNullable<T> utility removes null and undefined from a given type.

Example: Ensuring Non-Nullable Values#
type UserInput = string | null | undefined;
type ValidInput = NonNullable<UserInput>;

// Result: string

Use case: Preventing unexpected null values in function parameters or state management.

4. ReturnType<T> - Inferring Function Return Types#

The ReturnType<T> utility extracts the return type of a function.

Example: Extracting a Function's Return Type#
function getUser() {
  return { id: 1, name: "Alice" };
}

type User = ReturnType<typeof getUser>;

// Result: { id: number; name: string; }

Use case: Useful for ensuring function return types remain consistent across the codebase.

5. Parameters<T> - Extracting Function Parameters#

The Parameters<T> utility retrieves the types of parameters from a function type.

Example: Getting Function Parameter Types#
function logMessage(message: string, level: number) {}

type LogParams = Parameters<typeof logMessage>;

// Result: [string, number]

Use case: Helpful for creating wrappers around functions while maintaining type safety.

6. InstanceType<T> - Extracting Instance Types from Classes#

The InstanceType<T> utility extracts the type of an instance from a class constructor.

Example: Getting an Instance Type#
class User {
  constructor(public name: string, public age: number) {}
}

type UserInstance = InstanceType<typeof User>;

// Result: { name: string; age: number; }

Use case: Useful when working with class-based architecture, ensuring correct instance typing.

7. ThisParameterType<T> - Extracting the Type of this in Functions#

The ThisParameterType<T> utility extracts the type of this from a function.

Example: Extracting the this Type#
class Logger {
  prefix = "[LOG]";

  log(this: Logger, message: string) {
    console.log(`${this.prefix} ${message}`);
  }
}

type LoggerThis = ThisParameterType<typeof Logger.prototype.log>;

// Result: Logger

Use case: Useful for working with object-oriented patterns where this is used inside methods.

8. OmitThisParameter<T> - Removing the this Type from Functions#

The OmitThisParameter<T> utility removes the this type from a function type.

Example: Removing this for Standalone Functions#
class Logger {
  prefix = "[LOG]";

  log(this: Logger, message: string) {
    console.log(`${this.prefix} ${message}`);
  }
}

type LogFunction = OmitThisParameter<typeof Logger.prototype.log>;

// LogFunction is now (message: string) => void

Use case: Enables functions to be used independently from their original context.

Best Practices for Using Type Utilities#

  • Keep types maintainable - Overusing complex utilities can make code harder to read.
  • Use utilities to improve reusability - Extracting and modifying types ensures DRY (Don't Repeat Yourself) principles.
  • Combine utilities when needed - Many utilities work well together, such as NonNullable<ReturnType<T>>.
  • Validate inferred types - Use tools like tsc or inline TypeScript Playground to check inferred types.

Conclusion#

TypeScript's lesser-known type utilities can significantly enhance code quality by improving type safety, maintainability, and flexibility. Understanding and applying these utilities can optimize your TypeScript code while reducing potential bugs.

By incorporating these utilities into your workflow, you can write more concise and efficient TypeScript applications.