记录:非常典型的一个redux问题

在 ApproveCenterList 组件里面,当 tab 的 activeKey 发生变化,我在 useEffect 里面调用 setType 去修改 redux 里面的 type 状态,当type发生变化以后,会去调用redux 的异步方法 getData,然后我就发现在 getData里面,执行console.log(type, 9888); 此时的type会调用2次,第一次的type是老的,第二次是新的,导致最后页面出现异常。

但是在 ApproveCenterList 组件里面,我直接点击按钮,在onClick里面调用setType,去修改 redux 里面的 type 状态,当 type 发生变化以后,会去调用redux 的异步方法 getData,然后我就发现在 getData里面,执行console.log(type, 9888); 此时的type页也会调用2次,第一次和第二次的type都是新的,不存在老的状态出来。

这是为什么呢?

  1. useEffect 场景‌:

    • 当 tab 的 activeKey 变化触发 useEffect 时,setType 会发起 Redux 状态更新
    • Redux 状态更新是异步的,而 React 的 useEffect 依赖项变化会立即触发重新渲染
    • 在第一次 getData 调用时,Redux 的 type 可能还未完成更新,导致获取到旧值
    • 第二次调用则是更新完成后的新值
  2. onClick 场景‌:

    • 直接点击按钮时,setType 同步触发,Redux 会立即处理状态更新
    • 由于setTypegetData 几乎在同一事件循环中执行,是同步操作,getData 两次调用都能获取到最新值

解决方案

方案一:使用 useLayoutEffect

scss 复制代码
jsxCopy Code
useLayoutEffect(() => {
  setType(newType);
}, [activeKey]);

这会确保在 DOM 更新前同步执行状态更新5。

方案二:在 getData 中添加延迟

javascript 复制代码
jsxCopy Code
const getData = () => {
  .....
  //yield delay(10);
  yield new Promise(resolve => setTimeout(resolve, 0));
  //延迟以后
  const currentType = store.getState().type;
  console.log(currentType, 9888);
  ....
  // 后续操作
}

根本原因

这种现象源于 React 的批量更新机制和 Redux 的中间件处理流程:

  1. useEffect 的依赖变化会触发重新渲染,此时 Redux 状态可能处于中间状态5
  2. Redux-saga 等中间件处理异步操作时会有微任务延迟
  3. 直接事件处理(如 onClick)会立即触发同步更新流程3

建议优先采用方案四的路由参数同步方案,这既能保持状态一致性,又能解决页面刷新后的状态恢复问题。

相关推荐
layman05282 小时前
webpack5 css-loader:从基础到原理
前端·css·webpack
半桔2 小时前
【前端小站】CSS 样式美学:从基础语法到界面精筑的实战宝典
前端·css·html
AI老李2 小时前
PostCSS完全指南:功能/配置/插件/SourceMap/AST/插件开发/自定义语法
前端·javascript·postcss
_OP_CHEN2 小时前
【前端开发之CSS】(一)初识 CSS:网页化妆术的终极指南,新手也能轻松拿捏页面美化!
前端·css·html·网页开发·样式表·界面美化
啊哈一半醒2 小时前
CSS 主流布局
前端·css·css布局·标准流 浮动 定位·flex grid 响应式布局
PHP武器库2 小时前
ULUI:不止于按钮和菜单,一个专注于“业务组件”的纯 CSS 框架
前端·css
电商API_180079052472 小时前
第三方淘宝商品详情 API 全维度调用指南:从技术对接到生产落地
java·大数据·前端·数据库·人工智能·网络爬虫
晓晓莺歌2 小时前
vue3某一个路由切换,导致所有路由页面均变成空白页
前端·vue.js
Up九五小庞2 小时前
开源埋点分析平台 ClkLog 本地部署 + Web JS 埋点测试实战--九五小庞
前端·javascript·开源
qq_177767373 小时前
React Native鸿蒙跨平台数据使用监控应用技术,通过setInterval每5秒更新一次数据使用情况和套餐使用情况,模拟了真实应用中的数据监控场景
开发语言·前端·javascript·react native·react.js·ecmascript·harmonyos