dev-resources.site
for different kinds of informations.
Why React Needs Keys, Why It Matters
Everyone uses React library all know that whenever we use a map to render, we need to pass the key. Otherwise, React will "yell" at us like this.
Oh nooo š±š±, we need to pass the key now and most of the time we will pass like this.
list.map((l, idx) => {
Ā Ā return (
Ā Ā Ā Ā <div key={idx}>
Ā Ā Ā Ā Ā Ā <span>{l}</span>
Ā Ā Ā Ā </div>
Ā Ā )
})
Problem solved ā , no more warning š
But...š¢š¢
Life is not a dream
Let jump into an example of why "Like is not a dream"
Our manager gives us a task to create a dynamic Form with multiple Input fields, the user is able to enter their information and allow us to add or delete Input.
So
- We already know how to render the map in React library ā
- We already know how to use useState in React Hook with an array ā
- We also know method push/splice on array ā
Easy one lah š...
Without the beat, we set up our application
export default function App() {
const [list, setList] = useState([]);
const handleDelete = (idx) => {
Ā Ā list.splice(idx, 1);
Ā Ā setList([...list]);
};
const handleAdd = () => {
list.push(`Information ${list.length + 1}`);
Ā Ā setList([...list]);
};
const handleChange = (idx, l) => {
Ā Ā list[idx] = l;
Ā Ā setList([...list]);
};
return (
Ā Ā <div className="App">
Ā Ā Ā Ā {list.map((l, idx) => {
Ā Ā Ā Ā Ā Ā return (
Ā Ā Ā Ā Ā Ā Ā Ā <div key={idx}>
Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā {FancyComponent(l, handleChange, idx)}
Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā <button onClick={() => handleDelete(idx)}>Delete</button>
Ā Ā Ā Ā Ā Ā Ā Ā </div>
Ā Ā Ā Ā Ā Ā );
Ā Ā Ā Ā })}
Ā Ā Ā Ā <button onClick={() => handleAdd()}>Add</button>
Ā Ā </div>
Ā Ā );
}
const FancyComponent = (l, handleChange, idx) => {
Ā Ā const onChange = (e) => {
Ā Ā Ā Ā e.preventDefault();
Ā Ā Ā Ā handleChange(idx, e.currentTarget.value);
Ā Ā };
Ā Ā
Ā Ā return <input defaultValue={l} onChange={(e) => onChange(e)} />;
}
Here is Playground
Done!!! Feel super awesome ššŖ, it works like charm. We pass to our manager, our manager like š¤©š
A few moment later, our manager comes back again. We thought he would asking for the beer to appreciate but he said he found a bug š±š. The deletion didn't work as expected š„ŗ
After he modified Information 3 to Information 323, deleted this input but the application deleted Information 4, Information 5
We like whattt š±š±š±, how it could be possible!!!, it just only 50 lines of codes...
After put the console.log everywhere, googling, StackOverflow... We found the problem is the index we passed before is changed it makes ReactJS confused and render incorrect.
And if we don't use idx anymore, use content instead. Problem solved! ā
<div className="App">
Ā Ā {list.map((l, idx) => {
Ā Ā Ā Ā return (
Ā Ā Ā Ā Ā Ā <div key={l}>
Ā Ā Ā Ā Ā Ā Ā Ā {FancyComponent(l, handleChange, idx)}
Ā Ā Ā Ā Ā Ā Ā Ā <button onClick={() => handleDelete(idx)}>Delete</button>
Ā Ā Ā Ā Ā Ā </div>
Ā Ā Ā Ā Ā );
Ā Ā Ā })}
Ā Ā <button onClick={() => handleAdd()}>Add</button>
</div<
PS: Still has anĀ issue, waiting for anyone to figure it out. š
So now you might wonder that it is React's bug?!. No, it is not React's bug. Let's deep dive into it
If you noticed that when we change the value of the input, it works perfectly but deleting didn't work well because "changing" and "moving" state it is very different. When "moving" React needs to know what key of component is which we passed as an index of the array and deleting is changing the index hence it makes ReactJS confused. So why React don't make the by itself.
Dan's explanation:
React canāt make up a good key itself. Only you know how your data is structured and whether two circles in two renders are āthe sameā circle conceptually (even if all of its data changed) or not. Usually, youād use an ID generated during data creation. Such as from a databaseReact canāt make up a good key itself. Only you know how your data is structured and whether two circles in two renders are āthe sameā circle conceptually (even if all of its data changed) or not. Usually, youād use an ID generated during data creation. Such as from a database
It is 100% true, as an example, we go through above that it is a business requirement, as library React doesn't know what the key should be which only us, developers created those components.
What happens if you pass a random key every time?
Well, youāre telling React that circles are never the same between renders. So if you have some state inside of them, it will be lost after every re-render. React will destroy and recreate every circle because you told it to.
We will lose the "beauty" of ReactJS, isn't it? I believe no one wants to re-render every single time unless you have special requirements.
You might wonder why this topic pops up now, the concept of Keys has been there for a long time. ReactJS core team has started to built-in deep support Animation so Keys will play a big role there and in the future.
Bonus:
- ReactJS Core will update documentation about keys for more details.
- You might have awake of ReactJS 18 Alpha out and you also can follow ReactJS 18 working group to follow what the changes and new APIs are.
- React 18: https://reactjs.org/blog/2021/06/08/the-plan-for-react-18.html
- React 18 Working Group: https://github.com/reactwg/react-18 Hope you enjoy the article š
The article is inspired by Dan Abramov https://twitter.com/dan_abramov/status/1415279090446204929
Featured ones: