[英]State Value Stops Updating Within UseEffect Hook - Gatsby/React
状态值showMenu
未在useEffect
挂钩内更新。
测试时,当第一次点击按钮并触摸屏幕移动时, showMenu
正确控制台为true
。 当第二次(以及第三次、第四showMenu
)点击按钮并触摸屏幕移动时, showMenu
继续控制台为true
时它应该交替为false
。
const [showMenu, setShowMenu] = useState(false)
useEffect(_ => {
const listener = e => {
e.preventDefault()
console.log(showMenu, ' useEffect - touchmove')
}
showMenu
? document.body.addEventListener('touchmove', listener, {passive: false})
: document.body.removeEventListener('touchmove', listener)
}, [showMenu])
return (
<button onclick={_ => {
console.log(!showMenu, ' button click')
setShowMenu(!showMenu)
}} />
)
控制台结果
我不知道您的意图是什么,但是您对useEffect
所做的事情可能不是您所期望的。 当showMenu
为false
,您将删除尚未绑定的listener
函数,因为对象在 JS 中通过引用进行比较,并且每次showMenu
更改时都会重新定义listener
器。
当useEffect
更改时解除绑定侦听器的典型方法是返回一个处理useEffect
回调中的清理的函数。 像这样:
useEffect(() => {
const listener = e => {
e.preventDefault()
console.log(showMenu, ' useEffect - touchmove')
}
document.body.addEventListener('touchmove', listener, { passive: false })
return () = {
document.body.removeEventListener('touchmove', listener, { passive: false })
}
}, [showMenu])
我认为 body 的事件没有正确删除,因为每次useEffect
都会更改侦听useEffect
。
所以你可以在useEffect
返回一个函数来清除之前的useEffect
。
useEffect(() => {
if (showMenu) {
const listener = e => {
e.preventDefault();
console.log(showMenu, ' useEffect - touchmove');
};
document.body.addEventListener('touchmove', listener, { passive: false });
return () => {
document.body.removeEventListener('touchmove', listener);
}
}
}, [showMenu]);
您还可以阅读清理效果以了解更多信息
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.