React组件中监听函数获取不到最新的state

场景:在useEffect事件中,更新setState或者redux后获取到的state值为第一次的值,不是最新的

例如:

export default function Travel(props) {
    const divRef = useRef(null)
    const [count, setCount] = useState(0)
    const handleClick = () => {
        console.log('click',count);
    }

    useEffect(() => {
        window.addEventListener('click', handleClick)
        return () => {
            window.removeEventListener('click', handleClick)
        }
    },[])
    console.log('out',count);


    return (
        <div ref={divRef} >
            <button onClick={() => {
        setCount(count => count + 1)

            }}>点击</button>
        </div>
    )
}

又或者

const [flag,setFlag] = useState(false);
  
  const onClick = ()=>{
    setFlag(!flag)
    console.log(flag)
  }
  
  useEffect(() => {
    document.addEventListener('click',onClick)
  }, []);

原因:

找了一些资料了解到

因为监听器绑定的是第一次render时生成的函数,这个函数的上下文中的state也是第一次的值,所以即便后面render了多次,因为绑定的是第一次render的函数,所以state值也是旧的。

解决方案1:

useEffect(() => {
    window.addEventListener('click', handleClick)
    return () => {
        window.removeEventListener('click', handleClick)
    }
},[count])

监听值的变化,绑定并解绑事件

解决方案2:

const stateRef = useRef(0)
const [state,setState] = useState(stateRef.current);

const Function = ()=>{
    let data = JSON.parse(JSON.stringify(stateRef));
    /*功能*/
    stateRef.current = newData;
    setState(stateRef.current)
}  

通过useRef创建变量来改变state的值

其他情况:

const stateRef = useRef(null);

useEffect(() => {
    if (
      !stateRef.current ||
      (reduxState &&
        JSON.stringify(reduxState) !== JSON.stringify(stateRef.current) &&
        JSON.stringify(reduxState.id) !== JSON.stringify(stateRef.current?.id))
    ) {
      sstateRef.current = JSON.parse(JSON.stringify(reduxState));
    }
  }, [reduxState]);

比方说redux缓存了一串数据,这个数据用于控制物体移动,就可以监听redux state,将值赋值给stateRef。需要注意防止重复赋值,不然会造成抖动。判断是否数据无变化,判断是否切换了redux的数据。

有其他优雅的方法,希望大佬们可以留言。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
React 函数组件,有两个主要的生命周期函数: 1. `useEffect` `useEffect` 可以看作是 `componentDidMount`、`componentDidUpdate` 和 `componentWillUnmount` 这三个生命周期函数的结合。 `useEffect` 接收一个函数作为参数,这个函数会在组件挂载后执行,也会在组件更新时执行(除非你传入了第二个参数并且这个参数的值没有变化)。如果这个函数返回了一个函数,则这个返回的函数会在组件卸载时执行。 例如: ```jsx import { useEffect } from 'react'; function MyComponent(props) { useEffect(() => { console.log('组件挂载后执行'); return () => { console.log('组件卸载时执行'); }; }, [props.someProp]); useEffect(() => { console.log('组件挂载后执行'); return () => { console.log('组件卸载时执行'); }; }, []); // ... } ``` 上面的代码,第一个 `useEffect` 会在组件挂载后执行,也会在 `props.someProp` 的值发生变化时执行清理函数;第二个 `useEffect` 会在组件挂载后执行,但不会对任何 props 进行监听,因此它只会在组件挂载时执行一次。 2. `useState` `useState` 是一个 Hook,用于在函数组件添加状态(state)。 使用 `useState` 可以声明一个状态变量和一个更新函数,例如: ```jsx import { useState } from 'react'; function MyComponent(props) { const [count, setCount] = useState(0); return ( <div> <p>你点击了 {count} 次</p> <button onClick={() => setCount(count + 1)}>点击我</button> </div> ); } ``` 上面的代码,我们使用 `useState` 声明了一个状态变量 `count` 和一个更新函数 `setCount`,并将 `count` 的初始值设为 0。在组件渲染时,我们可以使用 `count` 来渲染一个按钮和一段文字,当用户点击按钮时,我们可以使用 `setCount` 来更新 `count` 的值。由于 `count` 的值发生变化,因此 React 会重新渲染组件。 除了 `useEffect` 和 `useState`,还有一些其他的 Hook 可以用于函数组件的生命周期管理,例如 `useContext`、`useReducer`、`useCallback` 等等。这些 Hook 的使用方法和作用可以参考 React 文档。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值