Problem
Let's say you have to call an external API to submit a name change and API count number. Every time the name changes you have to call the remove name API and then call the add name API. Alongside this you need to count how many times the API was called regardless of which API you call and also send the count number to the API as well.
import React, { useEffect, useState } from "react";
export default function RefTest() {
const [text, setText] = useState("");
const [name, setName] = useState("");
const [cnt, setCnt] = useState(0);
// DOM handlers
const inputChangeHandler = ({ target }) => setText(target.value);
const sendHandler = () => setName(text);
// HOOK
useEffect(() => {
console.log(`API - Add name: ${name} cnt: ${cnt + 1}`);
setCnt(cnt + 1);
return () => {
console.log(`API - Remove name: ${name} cnt: ${cnt + 1}`);
setCnt(cnt + 1);
};
}, [name, setCnt]);
return (
<div>
<input type="text" value={text} onChange={inputChangeHandler} />
<button onClick={sendHandler}>Send</button>
<div>Name: {name}</div>
<div>Count: {cnt}</div>
</div>
);
}
Note: All these examples can be better coded but I am trying to demonstrate a scenario.
There are couple of issues in the code above:
-
ESLintissue where we have not addedcntas a dependency. - If you run the code the
cntis not correct because of closure it maintains an older value ofcntbefore it can increment.
Adding cnt as a dependency
Note: Please do not add cnt as dependency as it will cause an infinite render. But if you want to try, do it on a page which you can kill easily.
The main issue with this approach apart from the infinte render is that it's going to start calling the API even when the cnt changes. Which we don't want as we only want to call the API when name changes.
Solution
Maintain the cnt as a ref so that it can be updated and mutated without impacting the useEffect hook execution cycle.
import React, { useEffect, useState, useRef } from "react";
export default function RefTest() {
const [text, setText] = useState("");
const [name, setName] = useState("");
const [cnt, setCnt] = useState(0);
const cntRef = useRef(cnt);
// DOM handlers
const inputChangeHandler = ({ target }) => setText(target.value);
const sendHandler = () => setName(text);
// HOOKS
useEffect(() => {
console.log(`API - Add name: ${name} cnt: ${cntRef.current++}`);
setCnt(cntRef.current);
return () => {
console.log(`API - Remove name: ${name} cnt: ${cntRef.current++}`);
setCnt(cntRef.current);
};
}, [name, setCnt]);
return (
<div>
<input type="text" value={text} onChange={inputChangeHandler} />
<button onClick={sendHandler}>Send</button>
<div>Name: {name}</div>
<div>Count: {cnt}</div>
</div>
);
}
At this point I am using cnt in the state as well so that I can display it on UI otherwise it's not needed.
Conclusion
- Anytime you want the
useEffectto execute for stateS1but you want to use other state values inside it but don't want other states to trigger theuseEffectfor those states than useuseRefhook to store the other states. - This is particularly helpful if you subscribe to an API and in your handler you want to do something with the incoming data combined with other state data (not
S1) before handing it over to some other operation.
Top comments (4)
Thanks for the post!
I would suggest using the functional update form of setState if you don't have to access cnt inside
useEffect, e.g.console.log(cnt). It lets us specify how the state needs to change without referencing the current state (docs):This makes sense. I just used this code as an example to demonstrate the concept.. I will update the example later to be more specific. Like maybe if you had to send the count to the API call.
I see your point. π
Alternately, if you don't need to display the count, use a
useRefinstead and save yourself a re-render.