I Am
Volodymyr Hudyma
<FrontEndDeveloper />
You Are Here: Home/A Simple Guide To Generics In TypeScript

A Simple Guide To Generics In TypeScript

July 20, 2020

Table Of Contents

    One of the main tools in the toolbox for producing reusable code is generics.

    It allows the code to be used not only with a single type, but with a variety of types.

    So basically you are allowing users to use your code with their own types.

    The Problem

    Imagine the following situation: you build a Select component and it receives some props. One of these is options, which contains all available options that are displayed on the list.

    Due to the fact that you are using TypeScript, it is necessary to specify a type for each option element:

    type Option = {
      name: string;
      value: string;
    };

    So far the type looks good, but there is a potential problem: What if the value contains a number?

    Quick and dirty ways to fix this problem:

    type Option = {
      name: string;
      value: string | number;
    };
    
    // Or create a type alias
    type Value = string | number;
    
    type Option = {
      name: string;
      value: Value;
    };
    
    // Or use any (do not even consider that!)
    type Option = {
      name: string;
      value: any;
    };

    But this code causes the type to be lost on the way, because if we get an option (e.g. in the onChange handler), we cannot tell if the value is string or number.

    The Solution

    The solution is to use generics:

    type Option<T> = {
      name: string;
      value: T;
    };
    
    // Or, to allow only string or number
    type Value = string | number;
    
    type Option<T extends Value> = {
      name: string;
      value: T;
    };

    Now, to use the proper type for the value:

    const numberOptions: Option<number> = [
      {
        name: "One",
        value: 1,
      },
      {
        name: "Two",
        value: 2,
      },
    ];
    
    const stringOptions: Option<string> = [
      {
        name: "John",
        value: "john",
      },
      {
        name: "Doe",
        value: "doe",
      },
    ];

    As you may have noticed, we use type variable to retrieve a type from the user and use it as a value type.

    The Identity Function

    The identity function is a function that will return back whatever is passed in.

    An example of an identity function without generics:

    const identity = (argument: number): number => {
      return argument;
    };

    An example of an identity function with generics:

    const identity = <T>(argument: T): T => {
      return argument;
    };

    Once we have defined the identity function, we can use it in one of two ways:

    // Pass the type argument to the function
    const result = identity<number>(10);
    
    // Use type argument inference
    // Compiler sets the type automatically, based on the type of provided value
    const result = identity(10);

    Important note: while type argument inference can be a helpful tool to keep code short and readable, you may better need to explicitly pass the type arguments for more complex types, as the compiler may fail to guess the proper type.

    What if the identity function should receive an array and return its length?

    const identity = <T>(argument: T[]): T[] => {
      return argument.length;
    };
    
    // The second way
    const identity = <T>(argument: Array<T>): Array<T> => {
      return argument.length;
    };
    
    // The third way
    // We do not limit function to arrays only
    // But allow to pass types which have length property
    interface ILength {
      length: number;
    }
    
    const identity = <T extends ILength>(argument: T): T => {
      return argument.length;
    };

    More Complex Example

    You can declare a type parameter that is constrained by another type parameter.

    For example, let's create a function that retrieves a property value from an object by name.

    We want to make sure that we do not accidentally grab a property that does not exist, so we will make a restriction between the two types:

    const getProperty = <T, K extends keyof T>(obj: T, key: K): number => {
      return obj[key];
    }
    
    const x = { a: 1, b: 2, c: 3 };
    
    getProperty(x, "a"); // returns "1"
    getProperty(x, "e"); // error: Argument of type "e" isn't assignable to "a" | "b" | "c".

    Summary

    The main reason to use generics in TypeScript is to enable types to act as parameters.

    It helps us to produce more reusable and maintainable code without any type duplicates.

    Newsletter
    Receive all new posts directly to your e-mail
    No spam, only quality content twice a week
    Let me know what you think about this article
    Click here to write response...