Skip to content
Advertisement

React, setInterval behavior

JavaScript

Code Sample: https://codesandbox.io/s/romantic-wing-9yxw8?file=/src/App.tsx


The code has two buttons – Start and Stop.

  • Start calls a setInterval and saves interval id. Timer set to 1 second (1000 ms).

  • Stop calls a clearInterval on the interval id.

The interval id is declared outside the component.

The interval callback function increments a counter and appends a called message to the UI.

When I click on Start, I expect the counter to increment every second and a corresponding called message appended underneath the buttons.

What actually happens is that on clicking Start, the counter is incremented just once, and so is the called message.

If I click on Start again, the counter is incremented and subsequently reset back to its previous value.

If I keep clicking on Start, the counter keeps incrementing and resetting back to its previous value.

Can anyone explain this behavior?

Advertisement

Answer

You have closure on count value inside the interval’s callback. Therefore after the first state update with value setState(0+1), you will have the same count value call setState(0+1) that won’t trigger another render.

Use functional updates which uses the previous state value without closures:

JavaScript

Same reason for messages:

JavaScript
JavaScript

Edit tender-currying-vqi49


Notice for another possible bug using an outer scope variable instead of useRef, for this read about useRef vs variable differences.


For a reference, here is a simple counter toggle example:

JavaScript

Edit Counter Toggle

User contributions licensed under: CC BY-SA
8 People found this is helpful
Advertisement