State not updating when using React state hook within setInterval(在 setInterval 中使用 React 状态挂钩时状态未更新)
问题描述
我正在尝试新的 React Hooks 并有一个带有时钟组件计数器应该每秒增加一次.但是,该值不会增加超过 1.
I'm trying out the new React Hooks and have a Clock component with a counter which is supposed to increase every second. However, the value does not increase beyond one.
function Clock() {
const [time, setTime] = React.useState(0);
React.useEffect(() => {
const timer = window.setInterval(() => {
setTime(time + 1);
}, 1000);
return () => {
window.clearInterval(timer);
};
}, []);
return (
<div>Seconds: {time}</div>
);
}
ReactDOM.render(<Clock />, document.querySelector('#app'));
<script src="aHR0cHM6Ly91bnBrZy5jb20vcmVhY3RAMTYuNy4wLWFscGhhLjAvdW1kL3JlYWN0LmRldmVsb3BtZW50Lmpz"></script>
<script src="aHR0cHM6Ly91bnBrZy5jb20vcmVhY3QtZG9tQDE2LjcuMC1hbHBoYS4wL3VtZC9yZWFjdC1kb20uZGV2ZWxvcG1lbnQuanM="></script>
<div id="app"></div>
推荐答案
原因是因为传入setInterval
的闭包的回调只访问了time
变量第一次渲染,它无法在后续渲染中访问新的 time
值,因为第二次没有调用 useEffect()
.
The reason is because the callback passed into setInterval
's closure only accesses the time
variable in the first render, it doesn't have access to the new time
value in the subsequent render because the useEffect()
is not invoked the second time.
time
在 setInterval
回调中的值始终为 0.
time
always has the value of 0 within the setInterval
callback.
和你熟悉的setState
一样,状态钩子有两种形式:一种是接受更新的状态,另一种是传入当前状态的回调形式.你应该使用第二种在 setState
回调中形成并读取最新的状态值,以确保在递增之前获得最新的状态值.
Like the setState
you are familiar with, state hooks have two forms: one where it takes in the updated state, and the callback form which the current state is passed in. You should use the second form and read the latest state value within the setState
callback to ensure that you have the latest state value before incrementing it.
奖励:替代方法
Dan Abramov,在他的 setInterval 的主题hooks/" rel="noreferrer">博客文章 并提供解决此问题的替代方法.强烈推荐阅读!
Dan Abramov, goes in-depth into the topic about using setInterval
with hooks in his blog post and provides alternative ways around this issue. Highly recommend reading it!
function Clock() {
const [time, setTime] = React.useState(0);
React.useEffect(() => {
const timer = window.setInterval(() => {
setTime(prevTime => prevTime + 1); // <-- Change this line!
}, 1000);
return () => {
window.clearInterval(timer);
};
}, []);
return (
<div>Seconds: {time}</div>
);
}
ReactDOM.render(<Clock />, document.querySelector('#app'));
<script src="aHR0cHM6Ly91bnBrZy5jb20vcmVhY3RAMTYuNy4wLWFscGhhLjAvdW1kL3JlYWN0LmRldmVsb3BtZW50Lmpz"></script>
<script src="aHR0cHM6Ly91bnBrZy5jb20vcmVhY3QtZG9tQDE2LjcuMC1hbHBoYS4wL3VtZC9yZWFjdC1kb20uZGV2ZWxvcG1lbnQuanM="></script>
<div id="app"></div>
这篇关于在 setInterval 中使用 React 状态挂钩时状态未更新的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:在 setInterval 中使用 React 状态挂钩时状态未更新
- 我不能使用 json 使用 react 向我的 web api 发出 Post 请求 2022-01-01
- 是否可以将标志传递给 Gulp 以使其以不同的方式 2022-01-01
- 从原点悬停时触发 translateY() 2022-01-01
- 如何调试 CSS/Javascript 悬停问题 2022-01-01
- 为什么我的页面无法在 Github 上加载? 2022-01-01
- 在不使用循环的情况下查找数字数组中的一项 2022-01-01
- 如何显示带有换行符的文本标签? 2022-01-01
- 为什么悬停在委托事件处理程序中不起作用? 2022-01-01
- 使用 iframe URL 的 jQuery UI 对话框 2022-01-01
- 如何向 ipc 渲染器发送添加回调 2022-01-01