DEV Community

Ridha Mezrigui
Ridha Mezrigui

Posted on

React.js : debouncing and throttling

Introduction

In order to build a professional web application, optimization and performance are two important things you need to care about.

There are many tips and techniques used to increase the performance of a web application, such as Debouncing and Throttling.

When it comes to debouncing and throttling, developers often confuse.

During this blog, I will go throw these two techniques in details using react.js, but it is the same principle for vanilla JavaScript or any other JavaScript framework.

Debouncing

Before dive deep into debouncing, let's see a simple and normal example that implement a search box that allows users to search something without clicking any button.

function App() {

  const handleChange = e => {
    console.log('api call...')
  }

  return (
    <div className="App">
      <header className="App-header">
        <p> Search  </p>
        <input type='text' onChange={handleChange} />
      </header>
    </div>
  );
}
Enter fullscreen mode Exit fullscreen mode

The issue is that handleChange is very expensive, and this is bad to the server because it will receive many HTTP requests in the same time.

Example
To solve the problem, we need to use a debounce function.

Definition and implementation of a debounce function

A debounce function is called after a specific amount of time passes since its last call.

function debounce(fn, delay) {
    let timer
    return function (...args) {
      clearTimeout(timer)
      timer = setTimeout(()=>fn(...args), delay)
    }
Enter fullscreen mode Exit fullscreen mode

The idea is to define a high-order function called debounce takes as arguments a callback function and a delay in ms, then returns a new function that sets the timer to execute the callback after the timer done.
The secret here is that every call of the function returned from the debounce function will cancel the previous timer using cleartimeout(timer) and start a new timer.
With this approach, we can be sure that the callback function will be executed just once after the time that we passed as an argument in the last call.

Implement debounce function into our example

 <div className="App">
      <header className="App-header">
        <p> Search  </p>
        <input type='text' onChange={debounce(handleChange, 500)} />
      </header>
    </div>
Enter fullscreen mode Exit fullscreen mode

Result

result

Throttling

Let's assume that we have an event listener in our app to track the movement of the user mouse, then send data to a backend server to do some operations based on the location of the mouse.

const handleMouseMove = e => {
      //everytime the mouse moved this function will be invoked
      console.log('api call to do some operations...')
  }
 //event listener to track the movement of the mouse
  window.addEventListener('mousemove',handleMouseMove)
Enter fullscreen mode Exit fullscreen mode

If we stick with this solution, we will end up with a down backend server because it will receive a hundred of requests in short duration.

event listener example
1600 API calls in few seconds is very very bad 📛📛📛.
To fix this issue, we need to limit the number of API calls, and this kind of problems can be solved using a throttle function.

Definition and implementation of a throttle function

A throttle function is a mechanism to limit the number of calls of another function in a specific interval, any additional calls within the specified time interval will be ignored.

function throttle(fn, delay) {
    let run = false
    return function (...args) {
      if (!run) {
        fn(...args)
        run = true
        setTimeout( () => run = false, delay)
      }
    }
  }
Enter fullscreen mode Exit fullscreen mode

The throttle function accepts two arguments: fn, which is a function to throttle, and delay in ms of the throttling interval and returns a throttled function.

Implement throttle function into our example

const handleMouseMove = e => {
      //everytime the mouse moved this function will be invoked
      console.log('api call to do some operations...')
  }

  //event listener to track the movement of the mouse
  window.addEventListener('mousemove', throttle(handleMouseMove, 1000))
//1000ms => 1 second
Enter fullscreen mode Exit fullscreen mode

Result

throttling example

Conclusion

Debouncing and Throttling are two amazing techniques, they can increase the performance of your web application to another level.
Choosing one of them depends on the case.

GitHub repo: https://github.com/ridhamz/debouncing-throttling

Latest comments (2)

Collapse
 
shehjad_dev profile image
Shehjad Ali

Hey , Ridha, wonderful post btw. Got one question though.

What this (...args) ?? Seen some of this usage in various other places as well. What is it or what does it mean? Any link that might answer my queries?

Collapse
 
ridhamz profile image
Ridha Mezrigui