javascript - 如何在不重新触发 useEffect 的情况下访问 useEffect 中的状态?
问题描述
我需要添加一些与 React 之外的对象交互的事件处理程序(以 Google 地图为例)。在这个处理函数内部,我想访问一些可以发送给这个外部对象的状态。
如果我将状态作为依赖项传递给效果,它可以工作(我可以正确访问状态)但是每次状态更改时都会添加添加/删除处理程序。
如果我不将状态作为依赖项传递,则添加/删除处理程序会添加适当的次数(基本上一次),但状态永远不会更新(或更准确地说,处理程序无法提取最新状态) .
代码笔示例:
也许最好用 Codepen 解释: https ://codepen.io/cjke/pen/dyMbMYr?editors=0010
const App = () => {
const [n, setN] = React.useState(0);
React.useEffect(() => {
const os = document.getElementById('outside-react')
const handleMouseOver = () => {
// I know innerHTML isn't "react" - this is an example of interacting with an element outside of React
os.innerHTML = `N=${n}`
}
console.log('Add handler')
os.addEventListener('mouseover', handleMouseOver)
return () => {
console.log('Remove handler')
os.removeEventListener('mouseover', handleMouseOver)
}
}, []) // <-- I can change this to [n] and `n` can be accessed, but add/remove keeps getting invoked
return (
<div>
<button onClick={() => setN(n + 1)}>+</button>
</div>
);
};
ReactDOM.render(<App />, document.getElementById("root"));
概括
如果效果的 dep 列表[n]
是更新状态,但每次状态更改都会添加/删除添加/删除处理程序。如果效果的 dep 列表是[]
添加/删除处理程序,但状态始终为 0(初始状态)。
我想要两者的混合。访问状态,但只访问一次 useEffect(好像依赖项是[]
)。
编辑:进一步澄清
我知道如何使用生命周期方法解决它,但不确定如何使用 Hooks。
如果上面是一个类组件,它看起来像:
class App extends React.Component {
constructor(props) {
super(props)
this.state = { n: 0 };
}
handleMouseOver = () => {
const os = document.getElementById("outside-react");
os.innerHTML = `N=${this.state.n}`;
};
componentDidMount() {
console.log("Add handler");
const os = document.getElementById("outside-react");
os.addEventListener("mouseover", this.handleMouseOver);
}
componentWillUnmount() {
console.log("Remove handler");
const os = document.getElementById("outside-react");
os.removeEventListener("mouseover", handleMouseOver);
}
render() {
const { n } = this.state;
return (
<div>
<strong>Info:</strong> Click button to update N in state, then hover the
orange box. Open the console to see how frequently the handler is
added/removed
<br />
<button onClick={() => this.setState({ n: n + 1 })}>+</button>
<br />
state inside react: {n}
</div>
);
}
}
ReactDOM.render(<App />, document.getElementById("root"));
注意添加/删除处理程序如何只添加一次(显然忽略了 App 组件未卸载的事实),尽管状态发生了变化。
我正在寻找一种用钩子复制它的方法
解决方案
您可以使用可变引用将读取当前状态与效果依赖项分离:
const [n, setN] = useState(0);
const nRef = useRef(n); // define mutable ref
useEffect(() => { nRef.current = n }) // nRef is updated after each render
useEffect(() => {
const handleMouseOver = () => {
os.innerHTML = `N=${nRef.current}` // n always has latest state here
}
os.addEventListener('mouseover', handleMouseOver)
return () => { os.removeEventListener('mouseover', handleMouseOver) }
}, []) // no need to set dependencies
const App = () => {
const [n, setN] = React.useState(0);
const nRef = React.useRef(n); // define mutable ref
React.useEffect(() => { nRef.current = n }) // nRef.current is updated after each render
React.useEffect(() => {
const os = document.getElementById('outside-react')
const handleMouseOver = () => {
os.innerHTML = `N=${nRef.current}` // n always has latest state here
}
os.addEventListener('mouseover', handleMouseOver)
return () => { os.removeEventListener('mouseover', handleMouseOver) }
}, []) // no need to set dependencies
return (
<div>
<button onClick={() => setN(prev => prev + 1)}>+</button>
</div>
);
};
ReactDOM.render(<App />, document.getElementById("root"));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.13.0/umd/react.production.min.js" integrity="sha256-32Gmw5rBDXyMjg/73FgpukoTZdMrxuYW7tj8adbN8z4=" crossorigin="anonymous"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.13.0/umd/react-dom.production.min.js" integrity="sha256-bjQ42ac3EN0GqK40pC9gGi/YixvKyZ24qMP/9HiGW7w=" crossorigin="anonymous"></script>
<div id="root"></div>
<div id="outside-react">div</div>
<p>Update counter with + button, then mouseover the div to see recent counter state.</p>
事件监听器只会在挂载/卸载时添加/删除一次。n
可以在内部读取当前状态,useEffect
而无需将其设置为依赖项([]
deps),因此不会重新触发更改。
您可以将其useRef
视为函数组件和 Hooks 的可变实例变量。类组件中的等价物将是this
上下文 - 这就是为什么类组件示例this.state.n
中的 in总是返回最新状态并正常工作。handleMouseOver
Dan Abramov有一个很好的例子,展示了上面的模式setInterval
。该博客文章还说明了useCallback
在每次状态更改时读取/删除事件侦听器以及何时读取/删除的潜在问题。
其他有用的示例是(全局)事件处理程序,例如os.addEventListener
React 边缘的外部库/框架或与外部库/框架的集成。
注意: React 文档建议谨慎使用此模式。从我的角度来看,在您只需要“最新状态”的情况下,它是一个可行的替代方案——独立于 React 渲染周期更新。通过使用可变变量,我们打破了具有潜在陈旧闭包值的函数闭包范围。
独立于依赖项编写状态还有其他选择 - 您可以查看如何使用 useEffect 挂钩注册事件?了解更多信息。
推荐阅读
- java - 如何将单击的列表项值传递给另一个活动?
- unity3d - 检查游戏对象在当前相机上是否可见
- python - 我怎样才能使这个阶乘函数递归?
- ocaml - 将自定义类型转换为“选项”自定义类型
- function - 在谷歌表格中发送电子邮件后出现 onEdit 问题
- c# - 缩进在 C# 中使用 XDocument 编辑的 Xml 文件
- swift - 在 macOS 上呈现 Xcode 的构建成功警报等警报视图的正确方法是什么?
- ajax - fetch() 发布请求操作
- automation - 如何在 Word 365 文档中创建 AutoOpen 宏
- html - 使用缩小的浏览器窗口@media css 包装到底线