zl程序教程

您现在的位置是:首页 >  工具

当前栏目

小前端读源码 - React16.7.0(深入了解setState)

源码前端 深入 了解 setState React16.7
2023-06-13 09:11:05 时间

在之前我们已经阅读过了React在首次渲染时的逻辑和流程,下面是链接:

Lam:小前端读源码 - React16.7.0(渲染总结篇)

但是对于阅读React源码的角度来说还不够,在上面文章最后有提到的一些阅读计划,本篇文章将去阅读在我们触发setState的时候到底代码是如何执行的,中间会经过哪些流程。

我们先来带着问题去阅读代码,在这次阅读中我们带着以下的一些问题进行阅读,通过阅读源码弄清楚每一个问题。

  1. this.setState是从哪里来的?
  2. 为什么在短时间内连续setState两次甚至多次只会触发一次render?
  3. 为什么setState是异步的?

本文的源码阅读以及断点都是基于以下DOME代码进行:

class App extends React.Component {
    constructor() {
        super();
        this.state = {
            data: 1
        }
    }
    render() {
        console.log('---render App---')
        return (
            <div>
                <p>text</p>
                <button onClick={() => {this.setState({data: 2})}}>setState</button>
            </div>
        )
    }
}

ReactDOM.render(
    <App/>,
    document.getElementById('root')
)

this.setState是从哪里来的

在之前的文章中有说到beginWork这个函数,会对不同类型的组件进行不同处理最终返回出一个Fiber节点,每一个class类型的Fiber节点都会在beginWork这个函数中调用到updateClassComponent函数,而updateClassComponent会调用constructClassInstance函数,在constructClassInstance会将当前的class组件实例化出来(class组件就是App组件),因为App组件是继承于React.Component的。所以当实例化的时候,在React.Compoent的原型上的setState将会被App组件所继承。从而setState就是从这里来的。

*在这里不能不说一下updater这个属性,因为在setState中调用的就是updater中的enqueueSetState函数!enqueueSetState函数是从constructClassInstance函数中实例化了class后,执行了一个adoptClassInstance函数,在里面对实例的对象的updater进行了赋值,并且将当前实例的Fiber节点赋值到实例的_reactInternalFiber属性中,留到之后使用。

var classComponentUpdater = {
  isMounted: isMounted,
  enqueueSetState: function (inst, payload, callback) {
   ...
  },
  enqueueReplaceState: function (inst, payload, callback) {
    ...
  },
  enqueueForceUpdate: function (inst, callback) {
    ...
  }
};

setState批处理

关于为什么在短时间内setState多次只会触发一次render的问题,其实涉及面比较广,里面包含了一些合成事件(Synethic event)的一些问题,但是本文主要关注setState的内容,这里不会详细去说明合成事件的事情。

当我们点击button按钮触发onClick事件的时候,会通过合成事件分发对应的回调函数,执行onClick中的内容。在onClick函数中,我们进行了一次setState。执行了enqueueSetState函数。在函数内有几个重要的步骤:

  1. createUpdate:创建了一个update对象。
  2. enqueueUpdate:创建updateQueue对象,将update对象存入到当前Fiber节点的updateQueue对象中的firstUpdate和lastUpdate中。
  3. scheduleWork:调用requestWork函数。

requestWork函数中有一个很重要的代码,决定这次setState是否会批量处理。

如果这次的setState并不是由合成事件触发的,那么isBatchingUpdates将会为false。如果为false就会直接执行performSyncWork函数了,马上对这次setState进行diff和渲染了。

其中如果多次setState的话,enqueueUpdate函数会对多次setState所传入的state进行替换。

从上面的代码解析,也明白之前的两个问题:

  1. 为什么在短时间内连续setState两次甚至多次只会触发一次render?
  2. 为什么setState是异步的?

连续setState多次只触发一次render就是因为经过了合成事件的关系,合成事件先执行了onClick函数中的setState,修改了Fiber的updateQueue对象的任务,执行完onClick函数体后,再由合成事件让根Fiber进行渲染(当然这只是简化的说法而已)。所以无论你在一个事件内触发无数次setState,也只会触发一次render。

其实我们在生命周期内进行setState的话,也不会立马进行setState的,React的内部是有处理的,当React的组件还没有渲染完成的时候,isRendering是为true的。

为什么setState是异步呢?这也是因为刚刚说到的,合成事件会先执行onClick中的setState,但是并不会马上进行渲染,所以新的state只存在于Fiber节点的updateQueue中,并不会马上赋值到组件的state中。


怎么让setState马上执行,并不需要经过合成事件去处理呢?加个setTImeout试试!但是不建议,React这么做是有原因的,因为防止多次setState触发多次的render导致性能减低,所以我们的setState都应该保持在生命周期内或者合成事件内!(当然不是官方不建议的生命周期内哟)https://github.com/facebook/react/blob/c6bee765ba/packages/react-reconciler/src/ReactFiberScheduler.js#L1849