React setState 同步还是异步的?


prtyaa
prtyaa 2023-12-26 17:49:21 65501
分类专栏: 资讯

如果你接触过 React 开发,肯定会遇到 setState() 方法,它是在类组件中更新 state 的方式,也是更新 UI 的主要方法。

方法格式如下:

setState(updater, [callback])

根据 React 官方文档,不保证在调用此方法后 state 会立即更新:

Think of setState() as a request rather than an immediate command to update the component. For better perceived performance, React may delay it, and then update several components in a single pass. React does not guarantee that the state changes are applied immediately.

我们可以用一些例子探索不同场景下的表现。

例子 1. 在 componentDidMount 生命周期函数里使用 setState()

class App extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      count: 1
    };
  }
  componentDidMount() {
    this.setState({
      count: 2
    })
    console.log('count:', this.state.count); // count: 1
  }
  render() {
    return (
      <div className="App">
        Count: {this.state.count}
      </div>
    )
  }
}

在控制台看到 this.state.count 输出值为 1,并非刚设置的 2。所以在此情况下对 state 的更新是延迟执行的。

例子 2. 在合成事件监听器里使用 setState()

class App extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      count: 1
    };
  }
  hanldeClick = () => {
    console.log('before click, count:', this.state.count); // before click, count: 1
    this.setState({
      count: 3
    });
    console.log('after click, count:', this.state.count); // after click, count: 1
  }
  render() {
    return (
      <div className="App">
        Count: {this.state.count}
        <p><button onClick={this.hanldeClick}>click</button></p>
      </div>
    )
  }
}

handleClick 是一个监听点击事件的函数,在函数内部添日志发现在调用 setState() 前后直接访问 this.state.count 输出值并没有变化,都是原来的值 1。

所以此情况下对于 state 更新也是延迟执行的。

例子3. 在原生事件监听器中

class App extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      count: 1
    };
    this.btn = React.createRef();
  }
  hanldeClick = () => {
    console.log('before click, count:', this.state.count); // before click, count: 1
    this.setState({
      count: 3
    });
    console.log('after click, count:', this.state.count); // after click, count: 3
  }
  componentDidMount() {
    this.btn.current.addEventListener('click', this.hanldeClick);
  }
  render() {
    return (
      <div className="App">
        Count: {this.state.count}
        <p><button ref={this.btn}>click</button></p>
      </div>
    )
  }
}

上述代码中用了 addEventListener() 浏览器原生事件 API 为按钮注册了事件监听器,并在其中先打印出之前的 state,然后调用 setState() 并再次打印 state 值。点击按钮,输出显示在调用 setState() 后 state 值立即更新了。

例子4. 在 setTimeout 中

class App extends React.Component {
  constructor(props) {
    super(props);
    this.state = {
      count: 1
    };
  }
  componentDidMount() {
    setTimeout(() => {
      console.log('count:', this.state.count); // count: 1
      this.setState({
        count: 2
      });
      console.log('count:', this.state.count); // count: 2
    }, 0);
  }
  render() {
    return (
      <div className="App">
        Count: {this.state.count}
      </div>
    )
  }
}

在 setTimeout() 中使用 setState() 也会立即更新 state 值。

结论

setState 只是一个普通的函数,并没有同步/异步的说法。我们平时所说的同步/异步是指在调用它之后是否会立即更新对应的 state。

React 为了 DOM 渲染的性能可能会延迟 state 更新,所以表现出了 state 的“异步”更新效果。

对于 setTimeout,原生事件监听器等场景则会立即更新 state 值,表现出了“同步”的效果。

网站声明:如果转载,请联系本站管理员。否则一切后果自行承担。

本文链接:https://www.xckfsq.com/news/show.html?id=30938
赞同 0
评论 0 条
prtyaaL0
粉丝 1 发表 2554 + 关注 私信
上周热门
银河麒麟添加网络打印机时,出现“client-error-not-possible”错误提示  1448
银河麒麟打印带有图像的文档时出错  1365
银河麒麟添加打印机时,出现“server-error-internal-error”  1151
统信桌面专业版【如何查询系统安装时间】  1073
统信操作系统各版本介绍  1070
统信桌面专业版【全盘安装UOS系统】介绍  1028
麒麟系统也能完整体验微信啦!  984
统信【启动盘制作工具】使用介绍  627
统信桌面专业版【一个U盘做多个系统启动盘】的方法  575
信刻全自动档案蓝光光盘检测一体机  484
本周热议
我的信创开放社区兼职赚钱历程 40
今天你签到了吗? 27
信创开放社区邀请他人注册的具体步骤如下 15
如何玩转信创开放社区—从小白进阶到专家 15
方德桌面操作系统 14
我有15积分有什么用? 13
用抖音玩法闯信创开放社区——用平台宣传企业产品服务 13
如何让你先人一步获得悬赏问题信息?(创作者必看) 12
2024中国信创产业发展大会暨中国信息科技创新与应用博览会 9
中央国家机关政府采购中心:应当将CPU、操作系统符合安全可靠测评要求纳入采购需求 8

添加我为好友,拉您入交流群!

请使用微信扫一扫!