近来在学习react源码, 最初是直接从入口一行一行的看, 结果跟着调用的函数跳转来跳去头都晕了. 后来决定带着一个目的去看源码, 每次看只研究一个东西. 一开始最想了解的就是充满魔性的setState. 本文是我对setState的一些理解, 不当之处欢迎留言指正.

setState的魔性

看一下下边几个例子的输出情况.

例1 合成事件中的setState

import React from 'react';

export default class SetState extends React.Component {
constructor(props) {
super(props);
} state = {
count: 0
} click = () => {
this.setState({
count: this.state.count + 1,
})
console.log('count1', this.state.count);
this.setState({
count: this.state.count + 1,
});
console.log('count2', this.state.count);
} render() {
return (
<div onClick={this.click}>
count的值{this.state.count}
</div>
)
}
}
// 打印:
// count1 0
// count2 0

例2 生命周期函数中的setState

import React from 'react';

export default class SetState extends React.Component {
constructor(props) {
super(props);
} state = {
count: 0
} componentDidMount () {
this.setState({
count: this.state.count + 1,
})
console.log('count1', this.state.count);
this.setState({
count: this.state.count + 1,
});
console.log('count2', this.state.count);
} render() {
return (
<div>
count的值{this.state.count}
</div>
)
}
}
// 打印:
// count1 0
// count2 0

例3 setTimeout中的setState

import React from 'react';

export default class SetState extends React.Component {
constructor(props) {
super(props);
} state = {
count: 0
} componentDidMount () {
setTimeout(() => {
this.setState({
count: this.state.count + 1,
})
console.log('count1', this.state.count);
this.setState({
count: this.state.count + 1,
});
console.log('count2', this.state.count);
}, 0);
} render() {
return (
<div>
count的值{this.state.count}
</div>
)
}
}
// 打印:
// count1 1
// count2 2

例4 Promise中的setState

import React from 'react';

export default class SetState extends React.Component {
constructor(props) {
super(props);
} state = {
count: 0
} componentDidMount () {
Promise.resolve()
.then(() => {
this.setState({
count: this.state.count + 1,
})
console.log('count1', this.state.count);
this.setState({
count: this.state.count + 1,
});
console.log('count2', this.state.count);
})
} render() {
return (
<div>
count的值{this.state.count}
</div>
)
}
}
// 打印:
// count1 1
// count2 2

从例1和例2的输出结果来看, 在setState后直接取state的值发现并没有更新, setState对state的更新似乎是个异步的过程;

而从例3, 例4输出结果来看, setState又是一个同步更新state的操作, 可以立即拿到更新的结果.

也就是说, setState有的时候是异步的有的时候是同步的, 真是非常的魔性. 根据网上的一些文章和自己的实验可以得出如下结论.

  • 在合成事件, 生命周期函数中的setState是异步批量更新的, 不能立即拿到更新的结果, 多次setState只会走一次render
  • 在setTimeOut, setInterval, 原生事件, Promise中的setState是同步逐个更新的, 可以立即拿到更新的state, 而且每次setState都会走一次render

关于是批量更新还是非批量更新可以在render函数中打印查看

setState魔性表现揭秘

理解setState的异步批量更新

下边是个异步批量更新的示意图

这里将在合成事件, setTimeout等中的写的代码的调用称为Main Process.

例如下边componentDidMount中的代码的执行都叫Main process.

componentDidMount () {
this.setState({
count: this.state.count + 1,
});
console.log('count1', this.state.count);
this.setState({
count: this.state.count + 1,
});
console.log('count2', this.state.count);
}

直接结合这段代码分析上边的这个看起来很牛x的图.

首先执行一个setState, 判断是setState操作, 创建一个更新任务加入更新队列, 交给协调中心, 协调中心判断不需要更新, 继续执行main Process中的代码.

遇到第一个console, 直接执行, 打印时取出了state, 显然state没更新还是原来的值, 然后再执行Main Process代码.

遇到第二个setState, 注意此时取出的state是没有更新的, 再创建一个更新任务到更新队列, 交给协调中心, 协调中心判断不需要更新, 继续执行main Process中的代码. 然后执行了console, 取出的state是没更新的.

一定时间后, 协调中心再次调度, 发现可以更新了, 然后执行了更新队列的两个任务, 得到一个新的state, 然后更新this.state和视图.

从以上分析可以了解到为什么两个console打印的都是之前的值.

这里有一个黑盒, 协调中心怎么运行的, 这是以后需要研究的了, 目前尚不清楚, 可以猜测这里边应该有个setTimeout 或者类似setTimeout的东西.

理解setState的同步单个更新

下边是同步更新的示意图

这里还是结合一段代码来分析

import React from 'react';

export default class SetState extends React.Component {
constructor(props) {
super(props);
} state = {
count: 0
} click = () => {
setTimeout(() => {
this.setState({
count: this.state.count + 1,
})
console.log('count1', this.state.count);
this.setState({
count: this.state.count + 1,
});
console.log('count2', this.state.count);
}, 0);
} render() {
return (
<div onClick={this.click}>
count的值{this.state.count}
</div>
)
}
}

首先遇到第一个setState, 判断是setState, 创建一个更新任务到更新队列, 然后进入协调中心, 协调中心通过某种手段判断出需要同步更新, 直接执行更新队列的任务, 得到新的state, 然后更新视图, 继续执行Main Process中的代码.

遇到console, 直接执行, 取出state(注意是更新了的)答应.

然后又遇到setState(注意这里拿到的state是更新了的), 创建更新任务进入更新队列, 然后进入协调中心, 协调中心通过某种手段判断出需要同步更新, 直接执行更新队列的任务, 得到新的state, 然后更新视图, 继续执行Main Process中的代码.

再次遇到console, 直接执行, 取出state(注意是二次了的)答应.

从以上分析可以看出同步setState为什么是同步的, 原因就在于他没有一个异步判断过程, 直接更新了state.

几点待解决的问题

  • 协调中心是什么时候, 如何判断出需要更新的
  • 协调中心是如何识别是一个setState是在setTimeout还是在合成事件亦或生命周期等过程中的.

彩蛋

说一下阅读react源码的感受, 最开始直接看src目录, react部分还行, 比较容易.

但是到了react-dom就不行了, 各种调用, 各种乱七八糟的东西, 有时跟着函数调用跳来跳去, 结果最开始想干嘛的都忘了, 这样读起来真的很打击人.

其实读源码更多不是了解其代码组织方法, 而是了解核心原理.

下边是几个小建议:

  1. 带着问题读源码, 尤其是开始读的时候, 如果漫无目的的读, 会很没有成就感, 甚至是强烈的挫败感, 读了半天也不知道学到了什么
  2. react-dom 的src代码组织十分复杂, 建议直接读开发版的编译产物, 都在一个文件里, 比较容易找.
  3. 多用断点, 可以直接在开发版编译产物打断点看, 非常方便
  4. 不要纠结太多细节, 要抱有不求甚解的态度, 不懂的地方可以暂时放过

小结

setState是一个容易让人困惑的东西, 尤其对react初学者来说, 可能感觉有点琢磨不透. 本文结合源码和自己的理解对setState的同步异步机制做了一些分析. 有些地方可能并不是十分准确, 但希望能帮助对setState同步异步机制困惑的朋友理解一些其中的原理. 最后需要记忆一下什么场景是同步更新, 什么场景是异步更新, 这个是写代码能实实在在用的到的.

理解setState的更多相关文章

  1. react面试题——理解setState(源码object.assign)

    setState是异步的方式 this.setState({     counter:this.state.counter+1 }) console.log(this.state.counter) s ...

  2. 重谈react优势——react技术栈回顾

    react刚刚推出的时候,讲react优势搜索结果是几十页. 现在,react已经慢慢退火,该用用react技术栈的已经使用上,填过多少坑,加过多少班,血泪控诉也不下千文. 今天,再谈一遍react优 ...

  3. React-setState的那些事儿

    关于setState,使用过react的人应该再熟悉不过了,在hooks还不那么普及的时候,除了使用函数式组件,我们使用最多的应该就是类创建react的组件了,而在类组件中我们通常会使用state来管 ...

  4. 深入理解 React JS 中的 setState

    此文主要探讨了 React JS 中的 setState 背后的机制,供深入学习 React 研究之用. 在课程 React.js入门基础与案例开发 中,有些同学会发现 React JS 中的 set ...

  5. react中的setState的使用和深入理解

    前端框架从MVC过渡到MVVM.从DOM操作到数据驱动,一直在不断的进步着,提升着, angular中用的是watcher对象,vue是观察者模式,react就是state了,他们各有各的特点,没有好 ...

  6. 对于react中的this.setState的理解

    一.this.setState第二个参数的作用:修改数据更新后最新的DOM结构 二.this.setState为什么是异步的? 1.setState只在合成事件和钩子函数中是“异步”的,在原生事件和s ...

  7. React 的setState 理解

    我们都知道在React中,setState() 方法是用来改变组件状态的,在项目中也一直用,也没有出现什么问题(使用方法太简单了),但今天看了一篇文章,提到了setState 使用时的两个注意点,加深 ...

  8. react中this.setState的理解

    this.setState作用? 在react中要修改this.state要使用this.setState,因为this.state只是一个对象,单纯的修改state并不会触发ui更新.所以我们需要用 ...

  9. 多线程锁--怎么理解Condition

    在java.util.concurrent包中,有两个很特殊的工具类,Condition和ReentrantLock,使用过的人都知道,ReentrantLock(重入锁)是jdk的concurren ...

随机推荐

  1. 拼写单词[哈希表]----leetcode周赛150_1001

    题目描述: 给你一份『词汇表』(字符串数组) words 和一张『字母表』(字符串) chars. 假如你可以用 chars 中的『字母』(字符)拼写出 words 中的某个『单词』(字符串),那么我 ...

  2. 【实践总结】给Centos和Ubuntu设置静态网络IP以及配置ssh功能

    作为一名以Windows平台为主的开发者,在接触和使用Linux系统的过程中总会遇到一系列的问题.每当这时候,我相信大部分人是和我一样的处理办法,就是网上各种搜索尝试直到问题解决为止,而有些问题,前后 ...

  3. C#中的扩展方法(向已有类添加方法,但无需创建新的派生类型)

    C#中的扩展方法 扩展方法使你能够向现有类型"添加"方法,而无需创建新的派生类型.重新编译或以其他方式修改原始类型. 扩展方法是一种特殊的静态方法,但可以像扩展类型上的实例方法一样 ...

  4. 【翻译】无需安装Python,就可以在.NET里调用Python库

    原文地址:https://henon.wordpress.com/2019/06/05/using-python-libraries-in-net-without-a-python-installat ...

  5. 学习 Object-C: 简史

    对于一门语言的历史,我认为写一本书可能都不为过,关键是看你如何介绍和表达.当然每一个人的理解也大相径庭.本文阐述也仅仅只是冰山一角,如果需要深入了解,自己可能需要多花费一些心思. 这里也不会给大家说太 ...

  6. ZOJ3435

    题意略. 思路: 将每一个点的坐标 (x,y,z) 与 (1,1,1) 相减,得到向量 (x - 1,y - 1,z - 1) 我们实际上就是要求出 这样互质的三元组有多少对就行了. 我们把这个长方体 ...

  7. Leetcode之深度优先搜索&回溯专题-491. 递增子序列(Increasing Subsequences)

    Leetcode之深度优先搜索&回溯专题-491. 递增子序列(Increasing Subsequences) 深度优先搜索的解题详细介绍,点击 给定一个整型数组, 你的任务是找到所有该数组 ...

  8. 1.Sentinel源码分析—FlowRuleManager加载规则做了什么?

    最近我很好奇在RPC中限流熔断降级要怎么做,hystrix已经1年多没有更新了,感觉要被遗弃的感觉,那么我就把眼光聚焦到了阿里的Sentinel,顺便学习一下阿里的源代码. 这一章我主要讲的是Flow ...

  9. Java版SockeDemo案例,有很详细的注释

    一般是用一个线程池来处理接受到的请求 直接上代码(一) ServerThread层 import java.io.BufferedReader; import java.io.InputStreamR ...

  10. Navicat for mysql建立连接

    1. 安装Navicat for MySQL. 2. 点击连接->MySQL,打开SSH,填写主机名.端口.用户名.密码. 3. 连接->打开常规,设置连接名(可以自由指定).主机名.端口 ...