/react-js-interview-questions

A list of React Interview Questions that I personally I feel are enough to cover for all types of React based interviews.

What is the Virtual DOM?

The Virtual DOM (Document Object Model) is a concept used in web development frameworks, particularly popularized by React. It is a lightweight copy of the actual DOM, a representation of the structure of a web page. When changes are made to the data in a web application, the Virtual DOM is updated first rather than directly manipulating the real DOM.

A simplified process of how the Virtual DOM works:

Initial Render: The Virtual DOM is created based on the initial state of the application.

Updates: When the state of the application changes, a new Virtual DOM is created.

Diffing: The new Virtual DOM is compared with the previous one to identify the differences or changes (diffing).

Reconciliation: Only the differences are then applied to the actual DOM, minimizing the amount of manipulation needed and improving performance.

The Virtual DOM helps optimize the rendering process by reducing the number of direct manipulations on the real DOM, which is a more expensive operation.

What does create-react-app do?

It is a command line tool that helps create a boilerplate react project with the pre-configured development environment.

This includes the build setup using module bundlers like Webpack and a development server for testing the application locally.

It also sets up a testing environment using Jest and provides the necessary configurations for writing and executing tests for your React components.

This helps developers to get started quickly in building their applications instead of spending time on configuring setups for the build.

What's the purpose of binding in class components?

The binding step is necessary because class methods don’t automatically bind this to the class instance. Let’s demonstrate it with the help of the following ES6 class component:

class ExplainBindingsComponent extends Component {
  onClickMe() {
    console.log(this);
  }

  render() {
    return (
      <button onClick={this.onClickMe} type='button'>
        Click Me
      </button>
    );
  }
}

The component renders just fine, but when you click the button, you see undefined in your developer console log. This is one of the main sources of bugs developers encounter in React. If you want to access this.state in your class method, it cannot be retrieved because this is undefined. To make this accessible in your class methods, you have to bind the class methods to this. In the following class component the class method is properly bound in the class constructor:

class ExplainBindingsComponent extends Component {
  constructor() {
    super();

    this.onClickMe = this.onClickMe.bind(this);
  }

  onClickMe() {
    console.log(this);
  }

  render() {
    return (
      <button onClick={this.onClickMe} type='button'>
        Click Me
      </button>
    );
  }
}

Some things to avoid would be to stop binding in the render method. Because in that case, binding happens every time the render() method runs, meaning every time the component updates, which will hurt your application’s performance eventually. Binding the class method in the constructor need only be done once, when the component is instantiated.

Some developers will define the business logic of their class methods in the constructor. Avoid this approach as well, as it will clutter your constructor over time. The constructor is only there to instantiate your class with all its properties, so the business logic of class methods should be defined outside the constructor.

Class methods can be auto-bound using JavaScript ES6 arrow functions:

class ExplainBindingsComponent extends Component {
  onClickMe = () => {
    console.log(this);
  };

  render() {
    return (
      <button onClick={this.onClickMe} type='button'>
        Click Me
      </button>
    );
  }
}

Use this method if the repetitive binding in the constructor annoys you.

What's the difference between Controlled and UnControlled Components?

Controlled components get their data from a parent, using props. The state is typically handled by the parent component, and any changes to the state are passed down to the child component via props. On the other hand, Uncontrolled components do their own thing. They take care of their own data/state without relying on a parent to manage it.

Since the parent manages their state, controlled components are more predictable. Data flow is controlled and consistent, making it simpler to debug and trace issues. As changes in the component's state are initiated and directed by the parent. A downside with Controlled components would be that it involve more code to handle the flow of data between parent and child components.

What are Functional Stateless Components?

These components are stateless, meaning they don't manage local state within the component. They solely rely on the input provided through props to determine their behavior and render content. Functional Stateless Components are functions that take input and return an output. The inputs are the props, and the output is a component instance in plain JSX. They have no local state (stateless). You cannot access or update the state.

When does constructor method and render method runs in the context of class methods?

The constructor runs only once in the lifetime of a component, whereas the render() class method runs once at the beginning and every time the component updates.

What's the difference between Named & Default Exports?

In JavaScript ES6, you can import and export functionalities from modules. These can be functions, classes, components, constants, essentially anything you can assign to a variable. The act of exporting one or multiple variables is called a named export:

// file1
const firstname = 'Rabi';
const lastname = 'Siddique';

export { firstname, lastname };

And import them in another file with a relative path to the first file.

// file2.js
import { firstname, lastname } from './file1.js';

On the other hand, default statement can be used to export and import a single functionality.

// file1.js
const person = {
  firstname: 'Rabi',
  lastname: 'Siddique',
};

export default person;

You have to leave out the curly braces to import the default export. Also the import name can differ from the exported default name.

import rabi from './file1.js';

What is One-Way Data Flow in React and How it useful?

React follows one way data flow. That means I can pass data from parent component to the child components. But, I cannot pass data from child components to parent components.

If I have an issue in the parent component. I know that it's because of the parent component and not the child component. Because, the child components cannot mess with it's parent component. Basically, it makes debugging in React straight-forward.

What is TreeShaking?

Tree shaking is the optimization process that selectively includes only the essential, or live code in a final bundle. This technique ensures that the generated bundle contains only the code that is actively utilized in your project.

For example, I make the import:

import ReactDOM from 'react-dom';

In this initial import, the entire react-dom library is included. However, with tree shaking, you can optimize further by importing only the specific component you require:

import { createRoot } from 'react-dom';

This is better because when the final bundle is created, only the implemenation of createRoot is includeded. And not the entire ReactDOM.

Why Hooks should be called at the top level of your React function?

In React, Hooks should be called at the top of your functional component. We should not place them inside conditionals, loops or nested functions. But why? This is because React keeps track of the order by which these hooks are called. Using this order, React is able to maintain state values between re-renders. If you mix up the order of these steps, React gets confused, and it can't keep the data in order.

For example, consider this component:

function Form() {
  // 1. Use the name state variable
  const [name, setName] = useState('Mary');

  // 2. Use an effect for persisting the form
  useEffect(function persistForm() {
    localStorage.setItem('formData', name);
  });

  // 3. Use the surname state variable
  const [surname, setSurname] = useState('Poppins');

  // 4. Use an effect for updating the title
  useEffect(function updateTitle() {
    document.title = name + ' ' + surname;
  });

  // ...
}

How does React know which state corresponds to which useState call? The answer is that React relies on the order in which Hooks are called. Above example works because the order of the Hook calls is the same on every render:

// ------------
// First render
// ------------
useState('Mary'); // 1. Initialize the name state variable with 'Mary'
useEffect(persistForm); // 2. Add an effect for persisting the form
useState('Poppins'); // 3. Initialize the surname state variable with 'Poppins'
useEffect(updateTitle); // 4. Add an effect for updating the title

// -------------
// Second render
// -------------
useState('Mary'); // 1. Read the name state variable (argument is ignored)
useEffect(persistForm); // 2. Replace the effect for persisting the form
useState('Poppins'); // 3. Read the surname state variable (argument is ignored)
useEffect(updateTitle); // 4. Replace the effect for updating the title

// ...

As long as the order of the Hook calls is the same between renders, React can associate some local state with each of them. But what happens if we put a Hook call inside a condition?

// 🔴 We're breaking the first rule by using a Hook in a condition
if (name !== '') {
  useEffect(function persistForm() {
    localStorage.setItem('formData', name);
  });
}

The name !== '' condition is true on the first render, so we run this Hook. However, on the next render the user might clear the form, making the condition false. Now that we skip this Hook during rendering, the order of the Hook calls becomes different:

useState('Mary'); // 1. Read the name state variable (argument is ignored)
// useEffect(persistForm)  // 🔴 This Hook was skipped!
useState('Poppins'); // 🔴 2 (but was 3). Fail to read the surname state variable
useEffect(updateTitle); // 🔴 3 (but was 4). Fail to replace the effect

React wouldn’t know what to return for the second useState Hook call. React expected that the second Hook call in this component corresponds to the persistForm effect, just like during the previous render, but it doesn’t anymore. From that point, every next Hook call after the one we skipped would also shift by one, leading to bugs.

Why Local Variables are not suitable for states?

There are two reasons why local reasons are not perfect to be used as states.

  1. Betweeen re-renders the values of local variables are not preserved. It is set to its original value. This is because when React renders a component, it renders it from scratch.
  2. Local variables dont trigger renders. Due to which we can't reflect the values of local variables on the screen in case they change.

How does React handle multiple consecutive calls to the set function of the state in the same render cycle?

Within a render, the value of state remains the same. Also, when setting the state using set function, it changes it for the next render only. For example, consider this snippet:

import { useState } from 'react';

export default function Counter() {
  const [number, setNumber] = useState(0);

  return (
    <>
      <h1>{number}</h1>
      <button
        onClick={() => {
          setNumber(number + 1);
          setNumber(number + 1);
          setNumber(number + 1);
        }}>
        +3
      </button>
    </>
  );
}

In this snippet, upon clicking the button, you might think the state is being updated 3 times, so in the next render, the value might be 3. Which is not the case. React updates states only for the next render. Which means no matter how many times you call set function, it will update state based on the current value for the very next render. Which in this will be 1.

What are Refs in React?

Refs are useful when we want a component to remember information without triggering renders on value changes. You can use refs by importing the useRef hook and pass it an initial value:

import { useRef } from 'react';
const ref = useRef(0);

// Returns a plain object:

/*

{ 
  current: 0 // The value you passed to useRef
}

*/

We can access the current value of the ref using ref.current property. This value is intentionally mutable, meaning you can both read and write to it.

How Refs are different from State?

  • useRef(initialValue) returns { current: initialValue }. While useState(initialValue) returns the current value of a state variable and a state setter function ( [value, setValue]).

  • Refs doesn’t trigger re-render when you change it, unlike states.

  • Refs are mutable while state is immutable.

  • Refs are synchronous. State acts like a snapshot for every render and doesn’t update synchronously. But when you mutate the current value of a ref, it changes immediately:

ref.current = 5;
console.log(ref.current); // 5

When should we use Refs?

Refs are handy when you want to step outside of React and perhaps want to access to some external browser APIs. Some situations we can use refs:

  • Storing timeout IDs.
  • Manipulating and accessing DOM elements directly.

Although, React automaitcally updates the DOM as a result we don't need to do it ourselves. However, there are times when we need access to a DOM element, for example to focus an input field, or getting the size of an element or scroll an element. React does not have any inbuilt method to do it. For this purpose, we can use of refs. Here is an example:

import { useRef } from 'react';
const myRef = useRef(null);
<div ref={myRef}>

On the initial render, myRef contains null until when React creates the DOM and points myRef.current to the div containing it. We can now access this DOM node from our event handlers and use the built-in browser APIs defined on it.

What are Forward Refs in React?

You cannot attach refs to our own defined components. By default, React returns null in this case. And it is intentional behavior. You will in fact get this warning:

Warning: Function components cannot be given refs. Attempts to access this ref will fail. Did you mean to use React.forwardRef()?

This is because by default, React does not allow component to access DOM nodes of another component, even if its a child component. Manually manipulating another component’s DOM nodes makes your code even more fragile.

To make sure, a component provides access to its DOM nodes, we have to make use of forwardRef. A component can specify that it forwards its ref to one of its children. For example:

const MyInput = forwardRef((props, ref) => {
  return <input {...props} ref={ref} />;
});

How many steps are involved in any React Screen Update?

Any screen update in a React app happens in three steps:

  1. Trigger
  2. Render
  3. Commit

How to Trigger a render in React?

A render is trigger in React only two times. These are:

  1. Initial render of the component. It’s done by calling createRoot with the target DOM node, and then calling its render method with your component. Here is the code:
import Image from './Image.js';
import { createRoot } from 'react-dom/client';

const root = createRoot(document.getElementById('root'));
root.render(<Image />);

If we comment out this portion of the code, we will see a blank screen.

  1. A component's state is updated. When a component has been rendered initially we can trigger renders by updating state of the component using set function.

What is Rendering in React?

Rendering in React involves invoking functions. On the initial render its invoking the function of the root component. Subsequent renders invole invoking the function of the component whose state has been updated, along with any nested components.

During this rendering process, React assesses the properties of these components to identify changes. The goal is to determine the most efficient way to apply these changes to the actual DOM. However, this information is not immediately utilized; it is held until the commit phase.

What is the Commit Phase in React?

After rendering our components, React will modify the DOM now.

  1. For the initial render, React will use the appendChild() DOM API to put all the DOM nodes it has created on screen.
  2. For re-renders, React will apply the minimal necessary operations (calculated while rendering!) to make the DOM match the latest rendering output.

React only changes the DOM nodes if there’s a difference between renders. For example, Consider this component:

export default function Clock({ time }) {
  return (
    <>
      <h1>{time}</h1>
      <input />
    </>
  );
}

It re-renders with different props passed from its parent every second. Notice how you can add some text into the <input>, updating its value, but the text doesn’t disappear when the component re-renders. This works because during this last step, React only updates the content of <h1> with the new time. It sees that the <input> appears in the JSX in the same place as last time, so React doesn’t touch the <input> or its value.

What is Painting in React?

After rendering is done and React updated the DOM, the browser will repaint the screen. This process is known as browser rendering.

What are Effects in React?

Effects let you run some code after rendering so that you can synchronize your component with some system outside of React Effects run at the end of a commit after the screen updates. An example of an effect is connecting to Chat server after the app loads, in case of a chat app. Which can done using the useEffect hock:

import { useEffect } from 'react';

function MyComponent() {
  useEffect(() => {
    // Code here will run after *every* render
  });
  return <div />;
}

Every time your component renders, React will update the screen and then run the code inside useEffect. In other words, useEffect “delays” a piece of code from running until that render is reflected on the screen.

Summary of declaration of useEffect:

useEffect(() => {
  // This runs after every render
});

useEffect(() => {
  // This runs only on mount (when the component appears)
}, []);

useEffect(() => {
  // This runs on mount *and also* if either a or b have changed since the last render
}, [a, b]);

What are cleanup functions in useEffect?

Some Effects need to specify how to stop, undo, or clean up whatever they were doing. For example, connect needs disconnect, subscribe needs unsubscribe, and fetch needs either cancel or ignore. You do this by returning a cleanup function.

You’re writing a ChatRoom component that needs to connect to the chat server when it appears. You are given a createConnection() API that returns an object with connect() and disconnect() methods. Something like:

function createConnection() {
  // A real implementation would actually connect to the server
  return {
    connect() {
      console.log('✅ Connecting...');
    },
    disconnect() {
      console.log('❌ Disconnected.');
    },
  };
}

useEffect(() => {
  const connection = createConnection();
  connection.connect();
}, []);

Imagine the ChatRoom component is a part of a larger app with many different screens. The user starts their journey on the ChatRoom page. The component mounts and calls connection.connect(). Then imagine the user navigates to another screen—for example, to the Settings page. The ChatRoom component unmounts. Finally, the user clicks Back and ChatRoom mounts again. This would set up a second connection—but the first connection was never destroyed! As the user navigates across the app, the connections would keep piling up.

To fix the issue, return a cleanup function from your Effect:

useEffect(() => {
  const connection = createConnection();
  connection.connect();
  return () => {
    connection.disconnect();
  };
}, []);

React will call your cleanup function each time before the Effect runs again, and one final time when the component unmounts (gets removed). Let’s see what happens when the cleanup function is implemented:

Why does useEffect prints ✅ Connecting... twice in code example below?

Code Example:

function createConnection() {
  // A real implementation would actually connect to the server
  return {
    connect() {
      console.log('✅ Connecting...');
    },
    disconnect() {
      console.log('❌ Disconnected.');
    },
  };
}

useEffect(() => {
  const connection = createConnection();
  connection.connect();
}, []);

This Effect only runs on mount, so you might expect ✅ Connecting... to be printed once in the console. However, if you check the console, ✅ Connecting... gets printed twice. Why does it happen?

Seeing the ✅ Connecting... log twice helps you notice the real issue: your code doesn’t close the connection when the component unmounts. Bugs like this are easy to miss without extensive manual testing. To help you spot them quickly, in development React remounts every component once immediately after its initial mount.

To fix the issue, return a cleanup function from your Effect:

useEffect(() => {
  const connection = createConnection();
  connection.connect();
  return () => {
    connection.disconnect();
  };
}, []);

Let’s see what happens when the cleanup function is implemented. Now you get three console logs in development:

  1. ✅ Connecting...
  2. ❌ Disconnected.
  3. ✅ Connecting...

This is the correct behavior in development. By remounting your component, React verifies that navigating away and back would not break your code. Disconnecting and then connecting again is exactly what should happen! When you implement the cleanup well, there should be no user-visible difference between running the Effect once vs running it, cleaning it up, and running it again. There’s an extra connect/disconnect call pair because React is probing your code for bugs in development.

In production, you would only see ✅ Connecting... printed once. Remounting components only happens in development to help you find Effects that need cleanup. You can turn off Strict Mode to opt out of the development behavior.

How to stop Event Propagation in React?

By default, events bubbles up or propagates in React. It starts with where the event happened, and then goes up the tree.

export default function Toolbar() {
  return (
    <div
      className='Toolbar'
      onClick={() => {
        alert('You clicked on the toolbar!');
      }}>
      <button onClick={() => alert('Playing!')}>Play Movie</button>
      <button onClick={() => alert('Uploading!')}>Upload Image</button>
    </div>
  );
}

In the example above, no matter which button we click, two callbacks will be called. One for the button we clicked and one for the div. To stop this, we use the event object received by event handlers. By convention, it’s usually called e, which stands for event. You can use this object to read information about the event. That event object also lets you stop the propagation. If you want to prevent an event from reaching parent components, you need to call e.stopPropagation() like this: Button component does:

export default function Toolbar() {
  return (
    <div
      className='Toolbar'
      onClick={(e) => {
        e.stopPropagation();
        alert('You clicked on the toolbar!');
      }}>
      <button onClick={() => alert('Playing!')}>Play Movie</button>
      <button
        onClick={(e) => {
          e.stopPropagation();
          alert('Uploading!');
        }}>
        Upload Image
      </button>
    </div>
  );
}

How to prevent Page Reloads on Form Submission in React?

Some browser events have default behavior associated with them. For example, a <form> submit event, which happens when a button inside of it is clicked, will reload the whole page by default. You can call e.preventDefault() on the event object to stop this from happening:

export default function Signup() {
  return (
    <form
      onSubmit={(e) => {
        e.preventDefault();
        alert('Submitting!');
      }}>
      <input />
      <button>Send</button>
    </form>
  );
}

What is the difference between Functional and Class Based Components in React?

  1. Functional components utilize functions, whereas class-based components make use of ES6 classes.
  2. Since the advent of Hooks, functional components can use state through the useState hook. Conversely, class components manage local state using this.state.
  3. Class components have access to various lifecycle methods, such as componentDidMount and componentDidUpdate, which can be beneficial for specific tasks. Before the introduction of hooks, functional components lacked lifecycle methods. With hooks, they can now utilize lifecycle methods through useEffect.
  4. In terms of performance, class components entail slightly more overhead due to the additional complexity of class instantiation. However, in modern React applications with performance optimizations, the impact is often negligible.
  5. Functional components are generally considered more concise and readable. They are easier to understand and write, thanks to their reduced boilerplate code.

Why is it important to use the key attribute when using map in React?

In React, the key attribute is used when rendering a list of elements, typically with the map function, to help React identify which items have changed, been added, or been removed. The key should be a unique identifier for each element in the list.

React uses the key attribute to optimize the updating of the virtual DOM. When a list changes, React compares the new list with the previous one using keys. This allows React to efficiently update only the elements that have changed or are new, rather than re-rendering the entire list.

const myList = ['item1', 'item2', 'item3'];

const listItems = myList.map((item) => <li key={item}>{item}</li>);

What is JSX?

JSX is a syntax extension for JavaScript recommended by React. It looks similar to XML/HTML, but it allows you to write JavaScript code within your JavaScript files. JSX makes it more convenient to describe what the UI should look like. React elements are typically created using JSX. For example, the JSX code:

const element = <h1>Hello, JSX!</h1>;

is transpiled by Babel into the following JavaScript code:

const element = React.createElement('h1', null, 'Hello, JSX!');

The React.createElement function is a fundamental part of React and is used to create React elements. It takes the element type, props (attributes), and children as arguments.

What is the React Component Lifecycle?

The Three Main Phases in a component lifecycle are:

  1. Mounting Phase: This phase occurs when a component is initially created and inserted into the DOM.
  2. Updating Phase: This phase occurs when a component's state or props change, causing it to re-render.
  3. Unmounting Phase: This phase occurs when a component is removed from the DOM.

What are Higher Order Components in React?

Higher Order Components (HOCs) are a pattern in React, a popular JavaScript library for building user interfaces. They are functions that take a component and return a new component with enhanced functionality. The term "higher order" comes from functional programming, where functions can take other functions as arguments or return them as results. HOCs provide a way to reuse component logic, share code between components, and abstract common functionality. For example see this code:

import React, { useEffect, useState } from 'react';
import { Redirect } from 'react-router-dom';

// Higher Order Component for Authentication
const withAuth = (WrappedComponent) => {
  return (props) => {
    const [isAuthenticated, setIsAuthenticated] = useState(false);

    useEffect(() => {
      const checkAuthentication = async () => {
        // Logic for checking authentication
        const isAuthenticated = /* Check if user is authenticated */ true;
        setIsAuthenticated(isAuthenticated);
      };

      checkAuthentication();
    }, []);

    if (isAuthenticated) {
      // If authenticated, render the wrapped component
      return <WrappedComponent {...props} />;
    } else {
      // If not authenticated, redirect to the login page
      return <Redirect to='/login' />;
    }
  };
};

const HomePage = () => <div>Welcome to the Home Page!</div>;
const AuthenticatedHomePage = withAuth(HomePage);

The code defines a reusable function that can wrap around a React component. This wrapper function withAuth, checks whether a user is authenticated. If authenticated, it renders the original component; if not, it redirects to a login page.

What is the useMemo hook?

useMemo is a React Hook that lets you cache the result of a calculation between re-renders.

const cachedValue = useMemo(calculateValue, dependencies);

calculateValue is the function calculating the value that you want to cache. It should be pure, should take no arguments, and should return a value of any type. React will call your function during the initial render. On next renders, React will return the same value again if the dependencies have not changed since the last render. Otherwise, it will call calculateValue, return its result, and store it so it can be reused later.

For example:

import { useMemo } from 'react';

function TodoList({ todos, tab }) {
  const visibleTodos = useMemo(() => filterTodos(todos, tab), [todos, tab]);
}

What is Lifting State Up in React?

Lifting State Up is a pattern in React where the state is moved to a higher-level (parent) component from its child components. This is done to share state among sibling components or to allow a common ancestor to control the state of multiple components.

// Parent.js
import React, { useState } from 'react';
import ChildA from './ChildA';
import ChildB from './ChildB';

function Parent() {
  const [sharedState, setSharedState] = useState('');

  const handleStateChange = (newState) => {
    setSharedState(newState);
  };

  return (
    <div>
      <ChildA sharedState={sharedState} onStateChange={handleStateChange} />
      <ChildB sharedState={sharedState} />
    </div>
  );
}

// ChildA.js
import React from 'react';

function ChildA({ sharedState, onStateChange }) {
  const handleChange = (e) => {
    onStateChange(e.target.value);
  };

  return (
    <div>
      <input type='text' value={sharedState} onChange={handleChange} />
    </div>
  );
}

// ChildB.js
import React from 'react';

function ChildB({ sharedState }) {
  return (
    <div>
      <p>Shared State: {sharedState}</p>
    </div>
  );
}

What is children prop?

In React, the children prop is a special prop that can be used to pass components or elements as children to another component. It allows a parent component to include arbitrary children components or elements within its JSX. The content passed as children is then accessible within the parent component using props.children. This makes it flexible for creating reusable and compositional components.

What are Pure Components?

Pure Components in React are a specific type of React component that automatically implements the shouldComponentUpdate lifecycle method with a shallow prop and state comparison. This means that a Pure Component will not re-render unless the actual data passed to it has changed. Pure Components can help optimize performance by preventing unnecessary renders when the component's output would be the same.

What are portals in React?

Portals in React provide a way to render children components outside the DOM hierarchy of the parent component. Portals allow you to render components at a different DOM node than their parent, without affecting the parent's styles or behavior.

When should we use fragments compared to divs?

Fragments in React are a lightweight way to group multiple elements without introducing an additional wrapping div to the DOM. Using fragments (<>...</>) helps avoid unnecessary and potentially undesirable nesting of elements. This is particularly beneficial when you want to keep your DOM structure clean and minimal, especially in cases where a wrapping div might interfere with styles or cause unintended layout effects.

What are error boundaries in React?

Error boundaries are React components that catch JavaScript errors anywhere in their component tree and log those errors, display a fallback UI, and prevent the error from crashing the entire component tree. They are implemented using the componentDidCatch lifecycle method.

Example of an error boundary component:

class ErrorBoundary extends React.Component {
  constructor(props) {
    super(props);
    this.state = { hasError: false };
  }

  componentDidCatch(error, errorInfo) {
    this.setState({ hasError: true });
    // Log the error to an error reporting service
  }

  render() {
    if (this.state.hasError) {
      return <h1>Something went wrong.</h1>;
    }
    return this.props.children;
  }
}

Also, this is possible in class based components only at the moment. There is currently no way to write an error boundary as a function component. You can use react-error-boundary package though.

What is the useCallback Hook?

useCallback is a hook that returns a memoized callback. In the context of React, a callback is often used as an event handler or to pass data between components. The useCallback hook takes two arguments: a function and a dependency array. It will return a memoized version of the function that only changes if one of the dependencies has changed.

This hook ensures that the callback function is only re-created if its dependencies change. This can be useful if a component is re-rendered frequently, as it can help to prevent the function from being unnecessarily re-created.