DEV Community

Cover image for Mastering 'useMemo' in React with TypeScript: 5 Different Use-Cases for 'useMemo'
Kirubel Kinfe
Kirubel Kinfe

Posted on

Mastering 'useMemo' in React with TypeScript: 5 Different Use-Cases for 'useMemo'

In React, the useMemo hook is a powerful tool that allows you to optimize performance by memoizing the result of expensive computations. When used in conjunction with TypeScript, it provides type safety and helps prevent common runtime errors. In this article, we'll explore the various use cases of useMemo in React, with a focus on best practices for TypeScript projects.

Understanding useMemo
The useMemo hook is part of the React Hooks API and is designed to memoize the result of a computation. It takes two arguments: a function that computes a value and an array of dependencies. The hook returns a memoized version of the computed value, which remains consistent across renders as long as the dependencies do not change.

Here's a basic example:

import React, { useMemo } from 'react';

function ExpensiveComponent({ data }: { data: number[] }) {
  const sum = useMemo(() => {
    return data.reduce((acc, val) => acc + val, 0);
  }, [data]);

  return <p>Sum: {sum}</p>;
}
Enter fullscreen mode Exit fullscreen mode

In this example, the sum variable is computed using the reduce function on the data array. The result is memoized using useMemo, ensuring that it is recalculated only when the data array changes.

Use Cases for useMemo in React and TypeScript

1. Optimizing Performance
The primary use case for useMemo is to optimize performance by preventing unnecessary recalculations. When a value depends on some expensive computation or data fetching, you can use useMemo to memoize the result and avoid recomputing it on every render.

import React, { useMemo } from 'react';

function ExpensiveComponent({ data }: { data: number[] }) {
  const sum = useMemo(() => {
    return data.reduce((acc, val) => acc + val, 0);
  }, [data]);

  return <p>Sum: {sum}</p>;
}
Enter fullscreen mode Exit fullscreen mode

In this example, the sum value is calculated using reduce. By wrapping this calculation in useMemo and specifying data as a dependency, you ensure that the sum value is only recomputed when the data prop changes.

2. Memoizing Expensive Components
You can use useMemo to memoize the rendering of complex or expensive components. This is especially useful when rendering components based on certain conditions.

import React, { useMemo } from 'react';

function ConditionalRendering({ showDetails }: { showDetails: boolean }) {
  const detailsComponent = useMemo(() => {
    return showDetails ? <Details /> : null;
  }, [showDetails]);

  return (
    <div>
      <p>Some content</p>
      {detailsComponent}
    </div>
  );
}
Enter fullscreen mode Exit fullscreen mode

In this example, the detailsComponent is conditionally rendered based on the showDetails prop. By using useMemo, you ensure that the Details component is only created when showDetails changes.

3. Preventing Unnecessary Propagation of Values
useMemo can be used to prevent unnecessary re-renders of child components. When passing computed or derived values as props to child components, memoizing those values can avoid unnecessary updates in the child components.

import React, { useMemo } from 'react';

function ParentComponent({ data }: { data: number[] }) {
  const sum = useMemo(() => {
    return data.reduce((acc, val) => acc + val, 0);
  }, [data]);

  return <ChildComponent sum={sum} />;
}

function ChildComponent({ sum }: { sum: number }) {
  // Render using the memoized `sum` prop
  return <p>Sum: {sum}</p>;
}
Enter fullscreen mode Exit fullscreen mode

By memoizing the sum value in the parent component using useMemo, you ensure that the ChildComponent doesn't re-render unless the data prop changes.

4. Complex Computed State
When you have complex computed state that depends on multiple values, useMemo helps you manage the dependencies efficiently. It allows you to calculate the value once when any of the dependencies change.

import React, { useMemo, useState } from 'react';

function ComplexStateComponent({ value1, value2 }: { value1: number; value2: number }) {
  const complexValue = useMemo(() => {
    // Perform complex computation using value1 and value2
    return value1 * value2;
  }, [value1, value2]);

  return <p>Complex Value: {complexValue}</p>;
}
Enter fullscreen mode Exit fullscreen mode

In this example, complexValue is computed based on value1 and value2. The useMemo hook ensures that complexValue is recalculated only when either value1 or value2 changes.

5. Memoizing Functions
useMemo can also be used to memoize functions that don't depend on component state. This is particularly useful when passing callback functions as props to child components.

import React, { useMemo } from 'react';

function ParentComponent() {
  const handleClick = useMemo(() => {
    return () => {
      console.log('Button clicked');
    };
  }, []);

  return <ChildComponent onClick={handleClick} />;
}

function ChildComponent({ onClick }: { onClick: () => void }) {
  // Render using the memoized `onClick` prop
  return <button onClick={onClick}>Click Me</button>;
}
Enter fullscreen mode Exit fullscreen mode

By memoizing the handleClick function with an empty dependency array ([]), you ensure that it doesn't change between renders, preventing unnecessary re-renders of the ChildComponent.

Conclusion
The useMemo hook in React, when combined with TypeScript, is a valuable tool for optimizing performance and preventing unnecessary recalculations. Understanding its various use cases is essential for building efficient and maintainable React applications. Whether you're optimizing performance, memoizing expensive components, preventing unnecessary re-renders, managing complex state, or memoizing functions, useMemo is a versatile hook that plays a key role in enhancing the performance and responsiveness of your React components.

Top comments (0)