I can't make it not work well if I use index as the key. The only way is if I mutate the array and use index as the key. But since the docs says not to mutate the state (the array), so if that's the case, I can't make it not work well, contrary to what the docs is stating. How can I show that it may break?
function App() {
const [arr, setArr] = React.useState(["A","B","C"]);
function toggleSortOrder() {
let newArr = [...arr];
newArr.reverse();
console.log(JSON.stringify(newArr));
setArr(newArr);
}
return (
<div>
<ul>
{ arr.map((e, i) => <li key={i}>{ e }</li>) }
</ul>
<button onClick={toggleSortOrder} >Toggle Sort Order</button>
</div>
)
}
ReactDOM.render(<App />, document.querySelector("#root"));
<script src="https://unpkg.com/[email protected]/umd/react.development.js" crossorigin></script>
<script src="https://unpkg.com/[email protected]/umd/react-dom.development.js" crossorigin></script>
<div id="root"></div>
I can make it break if I mutate the state, which the docs says should not be done:
function App() {
const [arr, setArr] = React.useState(["A","B","C"]);
function toggleSortOrder() {
arr.reverse();
console.log(JSON.stringify(arr));
setArr(arr);
}
return (
<div>
<ul>
{ arr.map((e, i) => <li key={i}>{ e }</li>) }
</ul>
<button onClick={toggleSortOrder} >Toggle Sort Order</button>
</div>
)
}
ReactDOM.render(<App />, document.querySelector("#root"));
<script src="https://unpkg.com/[email protected]/umd/react.development.js" crossorigin></script>
<script src="https://unpkg.com/[email protected]/umd/react-dom.development.js" crossorigin></script>
<div id="root"></div>
But I can't even break it if I mutate the state and use index as the key, if it is a class component:
class App extends React.Component {
state = { arr: ["A","B","C"] };
toggleSortOrder() {
this.state.arr.reverse();
console.log(JSON.stringify(this.state.arr));
this.setState({ arr: this.state.arr });
}
render() {
return (
<div>
<ul>
{ this.state.arr.map((e, i) => <li key={i}>{ e }</li>) }
</ul>
<button onClick={this.toggleSortOrder.bind(this)} >Toggle Sort Order</button>
</div>
);
}
}
ReactDOM.render(<App />, document.querySelector("#root"));
<script src="https://unpkg.com/[email protected]/umd/react.development.js" crossorigin></script>
<script src="https://unpkg.com/[email protected]/umd/react-dom.development.js" crossorigin></script>
<div id="root"></div>
The reality is that using the index of the
map
function will work, it is not recommended because it is an anti-pattern:If the array changes via add or removing an element, or modifying the number of elements in the array, React will assume that the DOM element represents the same component as before. Meaning the impact of using index could be unpredictable, such as two array items containing the same id.
It is therefore recommended that:
If your array elements contain a unique id, or you can form a unique id based on the structure, do so.
Use a global index such as a counter to help create dynamic ids... for example:
This way you ensure the
key
has a unique identifier.