react 定时器内闭包的存在导致 数据无法及时更新

需求:React Hooks useEffect使用定时器,每3秒更新一次值

代码如下:

javascript 复制代码
 const [MyV, setMyV] = useState(0);

  useEffect(() => {
    // 每隔3s,增加1
    const interval = setInterval(() => {
      setMyV(MyV+1);
    }, 3 * 1000);
    return () => {
      clearInterval(interval);
    };
  }, []);

实际,页面展示MyV 一直是 1。 因为setInterval会行程一个闭包。此闭包内用到的的上下文的state都是创建时的初始值。因此 setMyV(MyV+1); 时,MyV一直都是 0

优化方法1 :setMyV的函数形式

适用于此案例,只涉及一个state的变更

原理分析: setMyV((v) => v + 1); 里的v不属于闭包内部的变量,而是一个形参,因此不会一直使用0+1,而是最新的数据+1

javascript 复制代码
  const interval = setInterval(() => {
      setMyV((v) => v + 1);
    }, 3 * 1000);
优化方法2 :使用ref

适用于复杂案例,或许会涉及多个state的变更、甚至更多代码、方法、state的关联调用

原理分析:useRef 返回一个具有单个 current 属性 的 ref 对象,并初始化为你提供的 初始值。

且 在后续useRef 将返回相同的对象

因此,在此setInterval的闭包里,它只会关注myRef这个ref对象指向的地址 ,不再变化即可。myRef内部的current属性千变万化,以及myRef.current内部的其他属性用到的其它state (此处指MyV)也都是最新的。

javascript 复制代码
  const [MyV, setMyV] = useState(0);
  const myRef: any = useRef();
  myRef.current = () => {
    setMyV(MyV + 1);
    //只要是在这里访问、处理的,永远是最新的数据
    //...
  };
  useEffect(() => {
    // 每隔3s,增加1
    const interval = setInterval(() => {
      myRef.current();
    }, 3 * 1000);
    return () => {
      clearInterval(interval);
    };
  }, []);

参考:react官网对useRef的描述

相关推荐
LYFlied1 分钟前
浅谈跨端开发:大前端时代的融合之道
前端·flutter·react native·webview·大前端·跨端开发·hybrid
LYFlied1 分钟前
浅谈前端构建工具核心理解&&主流工具对比
前端·webpack·软件构建·rollup·vite·开发工具·工程化
weixin_3077791315 分钟前
Jenkins jQuery3 API 插件详解:赋能插件前端开发的利器
运维·开发语言·前端·jenkins·jquery
LinDon_24 分钟前
【企业微信快速登录适配 Chrome/Edge 142+】
前端·chrome·企业微信
JosieBook31 分钟前
【Vue】google chrome中安装vue_dev_tools.crx的时候提示“无法安装扩展程序,因为它使用了不受支持的清单版本。”
前端·vue.js·chrome
前端不太难34 分钟前
RN 性能优化:列表滚动掉帧、卡顿怎么办?
前端·react native·性能优化
亿元程序员34 分钟前
祖传项目二开快上线了,却还有很多旧的资源,怎么办?
前端
GIS学姐嘉欣37 分钟前
0帧起手《Vue零基础教程》,从前端框架到GIS开发
前端·vue.js·前端框架·gis
你说啥名字好呢37 分钟前
【React中的闭包陷阱】
javascript·react.js·ecmascript
麦麦在写代码39 分钟前
前端学习6(JS 1)
前端·javascript·学习