总结React中的setState
react中setState方法到底是异步还是同步,其实这个是分在什么条件下是异步或者同步。
1.先来回顾一下react组件中改变state的几种方式:
import React, { Component } from 'react'class Index extends Component { state={ count:1 } test1 = () => { // 通过回调函数的形式 this.setState((state,props)=>({ count:state.count+1 })); console.log('test1 setState()之后',this.state.count); } test2 = () => { // 通过对象的方式(注意:此方法多次设置会合并且只调用一次!) this.setState({ count:this.state.count+1 }); console.log('test2 setState()之后',this.state.count); } test3 = () => { // 不能直接修改state的值,此方法强烈不建议!!!因为不会触发重新render this.state.count += 1; } test4 = () => { // 在第二个callback拿到更新后的state this.setState({ count:this.state.count+1 },()=>{// 在状态更新且页面更新(render)后执行 console.log('test4 setState()之后',this.state.count); }); } render() { console.log('render'); return ( <p> <h1>currentState:{this.state.count}</h1> <button onClick={this.test1}>测试1</button> <button onClick={this.test2}>测试2</button> <button onClick={this.test3} style={{color:'red'}}>测试3</button> <button onClick={this.test4}>测试4</button> </p> ) } } export default Index;
2.setState()更新状态是异步还是同步:
需要判断执行setState的位置
同步:在react控制的回调函数中:生命周期钩子/react事件监听回调
import React, { Component } from 'react'class Index extends Component { state={ count:1 } /* react事件监听回调中,setState()是异步状态 */ update1 = () => { console.log('update1 setState()之前',this.state.count); this.setState((state,props)=>({ count:state.count+1 })); console.log('update1 setState()之后',this.state.count); } /* react生命周期钩子中,setState()是异步更新状态 */ componentDidMount() { console.log('componentDidMount setState()之前',this.state.count); this.setState((state,props)=>({ count:state.count+1 })); console.log('componentDidMount setState()之后',this.state.count); } render() { console.log('render'); return ( <p> <h1>currentState:{this.state.count}</h1> <button onClick={this.update1}>测试1</button> <button onClick={this.update2}>测试2</button> </p> ) } } export default Index;
异步:非react控制的异步回调函数中:定时器回调/原生事件监听回调/Promise
import React, { Component } from 'react'class Index extends Component { state={ count:1 } /* 定时器回调 */ update1 = () => { setTimeout(()=>{ console.log('setTimeout setState()之前',this.state.count);//1 this.setState((state,props)=>({ count:state.count+1 })); console.log('setTimeout setState()之后',this.state.count);//2 }); } /* 原生事件回调 */ update2 = () => { const h1 = this.refs.count; h1.onclick = () => { console.log('onClick setState()之前',this.state.count);//1 this.setState((state,props)=>({ count:state.count+1 })); console.log('onClick setState()之后',this.state.count);//2 } } /* Promise回调 */ update3 = () => { Promise.resolve().then(value=>{ console.log('Promise setState()之前',this.state.count);//1 this.setState((state,props)=>({ count:state.count+1 })); console.log('Promise setState()之后',this.state.count);//2 }); } render() { console.log('render'); return ( <p> <h1 ref='count'>currentState:{this.state.count}</h1> <button onClick={this.update1}>测试1</button> <button onClick={this.update2}>测试2</button> <button onClick={this.update3}>测试3</button> </p> ) } } export default Index;
3.setState()多次调用的问题:
异步的setState()
(1)多次调用,处理方法:
setState({}):合并更新一次状态,只调用一次render()更新界面,多次调用会合并为一个,后面的值会覆盖前面的值。
setState(fn):更新多次状态,只调用一次render()更新界面,多次调用不会合并为一个,后面的值会覆盖前面的值。
import React, { Component } from 'react'class Index extends Component { state={ count:1 } update1 = () => { console.log('update1 setState()之前',this.state.count); this.setState((state,props)=>({ count:state.count+1 })); console.log('update1 setState()之后',this.state.count); console.log('update1 setState()之前2',this.state.count); this.setState((state,props)=>({ count:state.count+1 })); console.log('update1 setState()之后2',this.state.count); } update2 = () => { console.log('update2 setState()之前',this.state.count); this.setState({ count:this.state.count+1 }); console.log('update2 setState()之后',this.state.count); console.log('update2 setState()之前2',this.state.count); this.setState({ count:this.state.count+1 }); console.log('update2 setState()之后2',this.state.count); } update3 = () => { console.log('update3 setState()之前',this.state.count); this.setState({ count:this.state.count+1 }); console.log('update3 setState()之后',this.state.count); console.log('update3 setState()之前2',this.state.count); this.setState((state,props)=>({ count:state.count+1 }));// 这里需要注意setState传参为函数模式时,state会确保拿到的是最新的值 console.log('update3 setState()之后2',this.state.count); } update4 = () => { console.log('update4 setState()之前',this.state.count); this.setState((state,props)=>({ count:state.count+1 })); console.log('update4 setState()之后',this.state.count); console.log('update4 setState()之前2',this.state.count); this.setState({ count:this.state.count+1 });// 这里需要注意的是如果setState传参为对象且在最后,那么会与之前的setState合并 console.log('update4 setState()之后2',this.state.count); } render() { console.log('render'); return ( <p> <h1>currentState:{this.state.count}</h1> <button onClick={this.update1}>测试1</button> <button onClick={this.update2}>测试2</button> <button onClick={this.update3}>测试3</button> <button onClick={this.update4}>测试4</button> </p> ) } } export default Index;
(2)如何得到setState异步更新后的状态数据:
在setState()的callback回调函数中
4.react中常见的setState面试题(setState执行顺序)
import React, { Component } from 'react'// setState执行顺序class Index extends Component { state={ count:0 } componentDidMount() { this.setState({count:this.state.count+1}); this.setState({count:this.state.count+1}); console.log(this.state.count);// 2 => 0 this.setState(state=>({count:state.count+1})); this.setState(state=>({count:state.count+1})); console.log(this.state.count);// 3 => 0 setTimeout(() => { this.setState({count:this.state.count+1}); console.log('setTimeout',this.state.count);// 10 => 6 this.setState({count:this.state.count+1}); console.log('setTimeout',this.state.count);// 12 => 7 }); Promise.resolve().then(value=>{ this.setState({count:this.state.count+1}); console.log('Promise',this.state.count);// 6 => 4 this.setState({count:this.state.count+1}); console.log('Promise',this.state.count);// 8 => 5 }); } render() { console.log('render',this.state.count);// 1 => 0 // 4 => 3 // 5 => 4 // 7 => 5 // 9 => 6 // 11 => 7 return ( <p> <h1>currentState:{this.state.count}</h1> <button onClick={this.update1}>测试1</button> <button onClick={this.update2}>测试2</button> <button onClick={this.update3}>测试3</button> <button onClick={this.update4}>测试4</button> </p> ) } } export default Index;
总结:react中setState()更新状态的2种写法
1)setState(updater,[callback])
updater:为返回stateChange对象的函数:(state,props)=>stateChange,接收的state和props都保证为最新的
2)setState(stateChange,[callback])
stateChange为对象,callback是可选的回调函数,在状态更新且界面更新后才执行
注意:
对象是函数方式的简写方式
如果新状态不依赖于原状态,则使用对象方式;
如果新状态依赖于原状态,则使用函数方式;
如果需要在setState()后获取最新的状态数据,在第二个callback函数中获取
本文来自 js教程 栏目,欢迎学习!
以上是总结React中的setState的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

如何利用React和WebSocket构建实时聊天应用引言:随着互联网的快速发展,实时通讯越来越受到人们的关注。实时聊天应用已经成为现代社交和工作生活中不可或缺的一部分。本文将介绍如何利用React和WebSocket构建一个简单的实时聊天应用,并提供具体的代码示例。一、技术准备在开始构建实时聊天应用之前,我们需要准备以下技术和工具:React:一个用于构建

React前后端分离指南:如何实现前后端的解耦和独立部署,需要具体代码示例在当今的Web开发环境中,前后端分离已经成为一种趋势。通过将前端和后端代码分开,可以使得开发工作更加灵活、高效,并且方便进行团队协作。本文将介绍如何使用React实现前后端分离,从而实现解耦和独立部署的目标。首先,我们需要理解什么是前后端分离。传统的Web开发模式中,前端和后端是耦合在

如何利用React和Flask构建简单易用的网络应用引言:随着互联网的发展,网络应用的需求也越来越多样化和复杂化。为了满足用户对于易用性和性能的要求,使用现代化的技术栈来构建网络应用变得越来越重要。React和Flask是两种在前端和后端开发中非常受欢迎的框架,它们可以很好的结合在一起,用来构建简单易用的网络应用。本文将详细介绍如何利用React和Flask

如何利用React和RabbitMQ构建可靠的消息传递应用引言:现代化的应用程序需要支持可靠的消息传递,以实现实时更新和数据同步等功能。React是一种流行的JavaScript库,用于构建用户界面,而RabbitMQ是一种可靠的消息传递中间件。本文将介绍如何结合React和RabbitMQ构建可靠的消息传递应用,并提供具体的代码示例。RabbitMQ概述:

React响应式设计指南:如何实现自适应的前端布局效果随着移动设备的普及和用户对多屏幕体验的需求增加,响应式设计成为了现代前端开发的重要考量之一。而React作为目前最流行的前端框架之一,提供了丰富的工具和组件,能够帮助开发人员实现自适应的布局效果。本文将分享一些关于使用React实现响应式设计的指南和技巧,并提供具体的代码示例供参考。使用React的Fle

React代码调试指南:如何快速定位和解决前端bug引言:在开发React应用程序时,经常会遇到各种各样的bug,这些bug可能使应用程序崩溃或导致不正确的行为。因此,掌握调试技巧是每个React开发者必备的能力。本文将介绍一些定位和解决前端bug的实用技巧,并提供具体的代码示例,帮助读者快速定位和解决React应用程序中的bug。一、调试工具的选择:在Re

ReactRouter使用指南:如何实现前端路由控制随着单页应用的流行,前端路由成为了一个不可忽视的重要部分。ReactRouter作为React生态系统中最受欢迎的路由库,提供了丰富的功能和易用的API,使得前端路由的实现变得非常简单和灵活。本文将介绍ReactRouter的使用方法,并提供一些具体的代码示例。安装ReactRouter首先,我们需

如何利用React和GoogleBigQuery构建快速的数据分析应用引言:在当今信息爆炸的时代,数据分析已经成为了各个行业中不可或缺的环节。而其中,构建快速、高效的数据分析应用则成为了许多企业和个人追求的目标。本文将介绍如何利用React和GoogleBigQuery结合起来构建快速的数据分析应用,并提供详细的代码示例。一、概述React是一个用于构建
