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>
相关推荐
一起学开源4 小时前
一文读懂 ReAct 范式:让 AI Agent 真正学会“思考+行动“
java·javascript·react.js·ecmascript·react·alibaba·智能体开发
格桑阿sir16 小时前
14-大模型智能体开发工程师:ReAct推理-行动框架
ai·大模型·llm·agent·react·智能体·推理模型
超级战斗鸡17 小时前
React Context Menu — 轻量级零依赖右键菜单组件
react
Richown3 天前
用 Three.js + React 打造一个赛博朋克风格的 3D 作品集页面
区块链·react
qcx235 天前
【系统学AI】08 Plan-then-Execute范式:先想好再做,比ReAct强在哪
前端·人工智能·react.js·ai·react·plan execute
JaydenAI6 天前
[MAF预定义ChatClient中间件-02]FunctionInvokingChatClient——实现ReAct循环和人机交互的大功臣
ai·人机交互·agent·react·hitl·maf·chatclient中间件
花月C8 天前
LangGraph 状态机与 ReAct Agent
python·agent·react·langgragh
Richown8 天前
密码学入门:区块链中的密码学原理
区块链·react
Richown8 天前
Web3钱包:钱包集成与签名验证
区块链·react
Richown9 天前
跨链桥接:多链资产转移实现
区块链·react