解决useState状态异步更新的问题

浅谈 useState 状态的异步更新

疑惑

相信刚开始使用react函数组件的小伙伴也遇到过一个坑,就是 useState 更新状态是异步更新的,但是react 并没有提供关于这个问题的解决方案。那我们能否使用自己的方法来解决这个问题呢?答案肯定是可以的。

状态异步更新带来的问题

就拿一个比较常见的场景来说。在react项目中,我们想在关闭对话框后再去处理其他业务。但是 useState 的状态是异步更新的。我们通过setVisible 更新状态后,状态并没有立马更新,这也就说明对话框并没有关闭,这也就造成了我们后面的逻辑在对话框没关闭时就执行了,这并不是我们想要的结果。下面来看我是如何来巧妙的解决这个问题的。

问题示例

// App.tsx
import {useState} from "react"

export default ()=>{

const [num,setNum]=useState(0)

const add=()=>{
  console.log("更新前",num)
  setNum(num+1)
  console.log('更新后',num)
}

return(
<div className='App'>
 <p>{num}</p>
 <button onClick={add}>num++</button>
</div>
)
}

下面是上面组件运行结果:
运行截图

点击按钮后的运行结果:
在这里插入图片描述

当我们点击按钮时的打印结果:
打印截图

问题解决

类组件的解决方案

在类组件中,我们可以在 setState(newstate,callback) 第二个参数传一个回调来处理本次状态更新后的一些其他业务。但是在函数组件中我们如何来解决这个问题呢?来看以下方案,也是我们这篇文章主要想为大家解决的问题。

函数组件的解决方案

解决该问题使用到的 api有:useEffectPromise

1.在项目源码目录下创建文件夹 customHooks,然后在 customHooks/useCallbackState.ts中编写如下代码:

import { useState, useRef, useEffect } from 'react';

export default(initState: any)=>{
    const stateRef = useRef(null as any);
    const [state, setState] = useState(initState);

    useEffect(() => {
    stateRef.current && stateRef.current(state);
    }, [state]);

    return [
        state,
        (newState:typeof initState):Promise<typeof initState>=>
        new Promise(rel=>{stateRef.current=rel;setState(newState)}) 
    ];
}

2.在上面的 App.tsx中使用上面的自定义hook

import useCallbackState from "@/customHooks/useCallbackState"
const [num,setNum]=useCallbackState(0)

const add=()=>{
   console.log('更新前',num)
   setNum(num+1)
   .then((newNum:any)=>{
   console.log('更新后',newNum)
   // console.log(num)
  })
}

此时的运行结果如下:
打印截图

结尾

如果有小伙伴能有更好的解决方案,欢迎一起讨论和分享。感谢大家的观看。

  • 8
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
useStateReact的一个Hook函数,可以用来在函数组件中保存和更新状态。但是,useState函数有一个异步更新状态问题。 在React的渲染过程中,useState函数的更新状态异步的,而不是立即生效的。这是因为React会对多个状态更新进行合并,然后一次性进行批量更新,以提高性能。所以,当我们在一个函数组件中多次调用useState函数来更新状态时,实际上只会触发一次UI的重新渲染。 举个例子,假设我们在一个函数组件中调用了两次useState函数来更新状态: ```jsx function MyComponent() { const [count, setCount] = useState(0); function handleClick() { setCount(count + 1); console.log(count); // 输出的是更新前的count值 setCount(count + 1); console.log(count); // 依然输出的是更新前的count值 } return ( <div> <p>Count: {count}</p> <button onClick={handleClick}>Click me</button> </div> ); } ``` 在上述例子中,我们点击按钮时会调用handleClick函数,该函数会两次更新count状态。但是,由于useState更新状态异步的,所以在两次调用setCount后立即打印count的值,并没有得到更新后的值,而是之前的值。 要解决这个问题,可以使用函数式的setState方式来更新状态,而不是直接传入一个新的值。如下所示: ```jsx function handleClick() { setCount(prevCount => prevCount + 1); setCount(prevCount => prevCount + 1); } ``` 使用这种方式,React会保证每次更新状态都是基于前一次更新后的值,而不是基于更新前的值。 综上所述,useState函数存在异步更新状态问题,但可以通过使用函数式的setState方式来解决这个问题

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值