useRef能持久化是因为它返回的对象在组件实例的生命周期内始终保持同一引用,React通过将该对象绑定到组件的内部节点(如Fiber节点)实现跨渲染的持久存储,每次调用useRef都返回同一实例,确保.current值在多次渲染间不变且修改不触发重渲染。

useRef
.current
本质上,
useRef
current
useRef
useRef(initialValue)
{ current: initialValue }useRef
current
useRef
useRef
useRef
当你第一次在组件里调用
useRef(initialValue)
{ current: initialValue }useRef
接下来,无论你的组件因为什么原因(比如父组件重渲染、自身状态改变)而重新渲染,React在执行组件函数时,会再次遇到
useRef
useRef
正是这种“首次创建,后续复用”的策略,确保了
useRef
current
useState
useRef
useState
这俩兄弟在表面上看起来都是用来“存东西”的,但骨子里却完全不同,用错了地方可能就是灾难。
useState
useState
而
useRef
.current
选择时机:
选择useState
选择useRef
简单来说,如果数据的变化需要“告诉”用户界面,那就用
useState
useRef
useRef
useRef
一个很常见的场景是存储上一次的值(Previous Value)。假设你想知道一个
prop
state
useRef
function MyComponent({ value }) {
const prevValueRef = useRef(); // 创建一个ref来存储上一个值
useEffect(() => {
// 在effect中更新ref的current属性,因为它会在渲染之后执行
prevValueRef.current = value;
}, [value]); // 只有当value变化时才执行
// 现在你可以随时访问 prevValueRef.current 来获取上一个值了
const prevValue = prevValueRef.current;
// ... 你的组件逻辑
return (
<div>
<p>当前值: {value}</p>
<p>上一个值: {prevValue !== undefined ? prevValue : '无'}</p>
</div>
);
}这种模式在处理动画、数据比较或者某些复杂的副作用逻辑时非常有用。
另一个妙用是存储可变的实例或计时器ID。比如,你可能需要在组件挂载时启动一个
setInterval
setInterval
function TimerComponent() {
const intervalRef = useRef(null); // 用ref来存储计时器ID
useEffect(() => {
intervalRef.current = setInterval(() => {
console.log('计时器在运行...');
}, 1000);
return () => {
// 组件卸载时清除计时器
if (intervalRef.current) {
clearInterval(intervalRef.current);
}
};
}, []); // 空依赖数组表示只在挂载和卸载时执行
return <div>查看控制台,我正在计时...</div>;
}这种方式确保了计时器ID在组件的整个生命周期内都是可访问的,并且可以在清理函数中正确使用。
我个人还喜欢用
useRef
isMounted
function AsyncDataFetcher() {
const isMounted = useRef(false); // 存储组件是否挂载的标志
useEffect(() => {
isMounted.current = true; // 组件挂载时设置为true
// 模拟数据请求
const fetchData = async () => {
// ... 假设这里有一些异步操作
await new Promise(resolve => setTimeout(resolve, 2000));
if (isMounted.current) { // 在更新状态前检查组件是否仍然挂载
console.log("数据已获取,组件仍在挂载");
// setState(...)
} else {
console.log("数据已获取,但组件已卸载,不再更新状态");
}
};
fetchData();
return () => {
isMounted.current = false; // 组件卸载时设置为false
};
}, []);
return <div>模拟异步数据获取...</div>;
}这个
isMounted
useState
以上就是JS如何实现useRef?Ref的持久化的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号