react 子组件调用父组件方法,获取的数据不是最新值

目录

      • 原因
      • 解决方法
          • [方法一、去掉 useCallback, 这样每次父组件触发刷新,就会刷新子组件](#方法一、去掉 useCallback, 这样每次父组件触发刷新,就会刷新子组件)
          • [方法二、或者通过监听 val,val 值改变来刷新函数](#方法二、或者通过监听 val,val 值改变来刷新函数)
          • [方法三、在父组件中,把 val 作为 key 值,每次 val 变化强制触发更新](#方法三、在父组件中,把 val 作为 key 值,每次 val 变化强制触发更新)

出现问题的代码如下:

typescript 复制代码
const Parent: React.FC = () => {
  const [val, setVal] = useState(0);

  const onBtnsClick = () => {
    console.log(val);
  };

  return (
    <div>
      {val}
      <button onClick={() => setVal(val => val + 1)}>加一</button>
      <Child onClick={onBtnsClick} />
    </div>
  );
};

const Child: React.FC<{ onClick: () => void }> = ({ onClick }) => {
  // useCallback + 防抖
  const onBtnClick = useCallback(
    _.debounce(onClick, 1000, { leading: true, trailing: false }),
    []
  );
  return <button onClick={onBtnClick}>子组件</button>;
};

原因

父组件状态的变更没有引起该子组件中的onBtnClick重新生成,导致方法中的 val 为一开始传入的数值0

解决方法

想办法触发组件刷新,使onBtnsClick 中打印的 val 永远是最新的值

方法一、去掉 useCallback, 这样每次父组件触发刷新,就会刷新子组件
js 复制代码
// 如果点击的事件不会导致父组件刷新,从而刷新子组件
const onBtnClick = _.debounce(onClick, 1000, {
  leading: true,
  trailing: false,
});

但这种方式只适用于一种情况:点击事件的处理不会导致父组件刷新;

如果父组件刷新,就会导致子组件刷新,从而 debounce 又新建,导致防抖无效。

如果想父组件刷新,子组件不刷新,可以父组件函数onBtnsClickuseCallback包裹,Child组件用memo包裹

完整代码如下:

js 复制代码
const Test: React.FC = () => {
  const [val, setVal] = useState(0);
  const onBtnsClick = useCallback(() => {
    console.log(val);
    setVal(val => val + 2)
  }, [val]);

  return (
    <div>
      {val}
      <button onClick={() => setVal(val => val + 1)}>加一</button>
      <Child onClick={onBtnsClick} />
  );
};

const Child = memo(({ onClick }) => {
  const onBtnClick = _.debounce(onClick, 1000, {
    leading: true,
    trailing: false,
  });
  return <button onClick={onBtnClick}>子组件</button>;
});
方法二、或者通过监听 val,val 值改变来刷新函数
js 复制代码
const onBtnClick2 = useCallback(
  _.debounce(onClick, 1000, { leading: true, trailing: false }),
  [val]
);
方法三、在父组件中,把 val 作为 key 值,每次 val 变化强制触发更新

这个改动是最小的

js 复制代码
<Child key={val} onClick={onBtnsClick} />

// 或者
<div key={val}>
  <Child  onClick={onBtnsClick} />
</div>
相关推荐
前端不太难1 小时前
RN 版本升级、第三方库兼容、Android/iOS 崩溃(实战博文 — 从 0.63 升到 0.72)
android·ios·react
前端无涯19 小时前
TypeScript 完整学习指南:从基础到工程化实践
typescript·vue·react
至善迎风2 天前
React2Shell(CVE-2025-55182)漏洞服务器排查完整指南
网络安全·react·数据安全·漏洞·next·rsc·cve-2025-55182
打小就很皮...2 天前
前端 Word 导出:自定义页眉表格的实现方案
前端·word·react·页眉设置
相逢一笑与君行3 天前
css使用grid布局实现网格(表格),动态调整行高,列宽,整体缩放,插入行,列,删除行,列
前端·css·react
purpleseashell_Lili4 天前
如何学习 AG-UI 和 CopilotKit
javascript·typescript·react
七夜zippoe4 天前
基于ReAct框架的智能体构建实战 - 从原理到企业级应用
前端·javascript·react.js·llm·agent·react
渴望成为python大神的前端小菜鸟5 天前
react 面试题
前端·react.js·前端框架·react·面试题
酒尘&6 天前
Hook学习-上篇
前端·学习·react.js·前端框架·react
人肉推土机6 天前
基于Claude Skills思想构建可拓展Plan ReAct Agent系统
langchain·react·mcp·claude skills·plan agent