Using Throttling and Debouncing with React hooks
https://dev.to/pulkitnagpal/using-throttling-and-debouncing-with-react-hooks-57f1
Throttling and debouncing techniques has been in use for past many years in javascript.
In this post I’d like to share my knowledge on how we can use throttle and debounce functions with help of react hooks.
Consider below example with two routes /
and /count
rendering respective components.
1 | export default function App() { |
Throttling Example with useEffect
Suppose we need to subscribe a scroll event on Count
component on its mount and just increment the count on every scroll event.
Code without using throttle or debounce techniques will be like:
1 | function Count() { |
Suppose in practical applications you need to use throttle and wait for every 100ms before we execute increaseCount
. I have used the lodash throttle function for this example.
1 | function Count() { |
Wait, no need to hurry. It will work if you are at /count
route. The increaseCount
function will be throttled and will increase the count after 100ms of intervals.
But as you move to the /
route to render the Home
component and unmount the Count
component, and start scrolling on home page, you will notice a warning in console which warns about memory leak. This is probably because the scroll event was not cleaned properly.
The reason is _.throttle(increaseCount, 100)
is called again during unmount and returns another function which does not match that created during the mount stage.
What if we create a variable and store the throttled instance.
like this
1 | const throttledCount = _.throttle(increaseCount, 100); |
But it has problem too. The throttledCount
is created on every render, which is not at all required. This function should be initiated once which is possible inside the useEffect hook. As it will now be computed only once during mount.
1 | useEffect(() => { |
Debounce Example using useCallback or useRef
Above example is pretty simple. Let’s look at another example where there is an input field and you need to increment the count only after user stops typing for certain time. And there is text which is updated on every keystroke which re renders the component on every input.
Code with debounce:
1 | function Count() { |
This will not work. The count will increase for every keystroke. The reason behind is that on every render, a new debouncedCount
is created.
We have to store this debounced function such that it is initiated only once like that in useEffect in above example.
Here comes use of useCallback
.useCallback
will return a memoized version of the callback that only changes if one of the dependencies has changed - React docs
Replace
1 | const debouncedCount = _.debounce(increaseCount, 1000); |
with
1 | const debouncedCount = useCallback(_.debounce(increaseCount, 1000),[]); |
and it will work. Because this time the function is evaluated only once at the initial phase.
Or we can also use useRef
by doing this
1 | const debouncedCount = useRef(debounce(increaseCount, 1000)).current; |
One should always keep in mind that every render call of react functional component will lead to expiration of local variables and re-initiation unless you memoize them using hooks.