React 是在内部创建 prevState 的深拷贝还是浅拷贝?
Posted
技术标签:
【中文标题】React 是在内部创建 prevState 的深拷贝还是浅拷贝?【英文标题】:Does React create a deep or shallow copy of prevState internally? 【发布时间】:2020-07-23 06:01:39 【问题描述】:我刚开始编程,最近用 React 编写了我的第一个应用程序,虽然它确实有效,但我不确定我是否正确处理了我的内部状态。我的问题是 setState 方法是否创建了“prevState”的不可变深层副本?举个例子:
menuAddRecHandler = () =>
this.setState((prevState) =>
const updatedState = ...prevState;
const dayInd = updatedState.menu.recipes.findIndex(item => item.dayNum === updatedState.menuInternal.currentDay);
updatedState.menu.recipes[dayInd].recList.push(
name: prevState.menuInternal.menuRecSelect,
portions: prevState.menuInternal.menuNumPortInput
);
return updatedState;
)
在我的处理程序中,我将一个对象推送到 updatedState 对象的嵌套数组,该数组是由扩展运算符从 prevState 复制过来的。现在我知道扩展操作符只做一个浅拷贝,但是 setState 方法内部提供的 prevState 也是浅的,这是否意味着我实际上是通过调用这个处理程序直接改变我的状态?如果是这样,我该如何解决这个问题?谢谢。
【问题讨论】:
【参考方案1】:来自docs:
第一个参数是带有签名的更新函数:
(state, props) => stateChange
state 是对应用更改时的组件状态的引用。它不应该直接突变。相反,应该通过基于 state 和 props 的输入构建一个新对象来表示更改。
如何在不变异的情况下达到相同的结果?
menuAddRecHandler = () =>
this.setState((prevState) =>
return
...prevState,
menu:
...prevState.menu,
recipes: prevState.menu.recipes.map((item) =>
if(item.dayNum === prevState.menuInternal.currentDay)
return
...item,
recList: [
...item.recList,
name: prevState.menuInternal.menuRecSelect,
portions: prevState.menuInternal.menuNumPortInput
]
return item;
)
);
这显然很容易出错,其想法是减少更深的状态。 Redux 的文档offer some tips for normalizing state。
您还可以查看不可变状态库,例如 ImmutableJS 或 Immer
【讨论】:
更好的答案。 非常好的答案,这些链接肯定会对作者做出更可维护的状态有用。 谢谢。我想我现在将专注于学习 Redux 来清理我造成的烂摊子。 @Glogov 不,你不必学习 Redux,该链接仅用于规范化状态,它也可以应用于本地状态。有一个page in redux docs可以帮助你评估是否需要redux @Agney 我会调查一下,现在我正在阅读您提到的 Immer 库。用这个特定的应用程序解决我的问题似乎是一种不错的、不复杂的方法。谢谢你提出来。【参考方案2】:According to the docs根本不是复制品
state 是对应用更改时的组件状态的引用。它不应该直接变异。
制作深拷贝通常对性能不利,但如果必须这样做,您可以这样做:
const updatedState = JSON.parse(JSON.stringify(prevState));
【讨论】:
我明白了。您的方法应该可以解决问题,因为它是一个相对较小的应用程序。但是,我知道我一开始就不应该创建这样一个嵌套的内部状态? 是的,有一个非常深的状态使得它很难使用并且修改成本很高(您至少需要重新创建作为您正在修改的值的父对象的每个对象)【参考方案3】:您的示例中有很多“噪音”,所以我会保持简单:
state = name: `Dennis`, age: 89
this.setState((prevState) =>
return age: prevState.age + 1 ;
); // name: `Dennis`, age: 90
this.setState( name: `John` ); // name: `John`, age: 89
在类组件中,this.setState
浅合并与之前的状态,prevState
是实际的状态引用,即: name: "Dennis", age: 89
。
updater 函数接收到的 state 和 props 都保证是最新的。更新器的输出与状态浅合并。
注意不是the case with functional components(常见错误)。
与类组件中的 setState 方法不同,useState 不会自动合并更新对象。
【讨论】:
以上是关于React 是在内部创建 prevState 的深拷贝还是浅拷贝?的主要内容,如果未能解决你的问题,请参考以下文章
ReactJS - 新的 useState React 钩子中的 prevState?
使用 prevState 和 useState 钩子更新对象内部数组中的对象
功能组件的setState中的prevState到底是什么?
componentDidUpdate(prevProps, prevState, snapshot): prevProps 未定义