我最近开始学习web开发,这让我头疼。我做了一个简单的todo应用程序,你可以添加一个id和值的项目到数组中,并从数组中删除该项目。现在我希望每当网站刷新时,我希望项目都在那里。我做了一个测试数组,出于某种原因,当刷新时,它可以工作,但不适用于data
。
const [items, setItems] = useState([]);
const testarr = [{id: 1, value: "a"}, {id: 2, value: "b"}];
useEffect(() => {
const data = JSON.parse(localStorage.getItem("items"));
console.log(data);
// setItems(testarr);
setItems(data);
}, [])
useEffect(() => {
window.localStorage.setItem("items", JSON.stringify(items));
}, [items]);
我想到的是:
useEffect(() => {
const data = JSON.parse(localStorage.getItem("items"));
data.forEach(element => {
setItems(oldarr => [...oldarr, element]);
});
}, [])
问题是,如果我不断刷新站点的速度足够快,最终数组将再次变为空。我不明白,当刷新时,它记录data
就可以了,但当我想记录setItems(data)
时,它就不工作了。有人知道为什么它会这样吗?
3条答案
按热度按时间cig3rfwq1#
setState是一个异步函数,这意味着下面的代码片段将始终在项目从“setItems(data)”接收数据之前以空数组的形式运行。
也许,你可以在存储之前检查items是否不是空数组
9gm1akwq2#
请尝试不带窗口的localStorage对象
例如:
i7uq4tfw3#
问题
下面的
useEffect
在每次items
发生变化时都会运行,但在挂载时也会运行。在挂载时,items
等于[]
,即一开始给useState
的数组。这就是为什么localStorage
被设置为[]
。溶液
解决此问题的一种方法是如下所示更改状态定义,因此当
localStorage
中有数据时,您可以选择它:此时,您可以删除用于从
localStorage
获取数据并将其提供给setItems
的两个useEffect
。