【Under-the-hood-ReactJS-Part9】React源码解读

接上文,

React流程图:
https://bogdan-lyashenko.gith...

回到最初

在流程图中,也许你已经注意到,setState方法可以通过几种方式触发,更准确的说,可以分为是否由外部引起的(也就是是否由用户触发)。让我们看下如下两个列子,第一个中,
是由鼠标点击触发的,第二个中,是在componentDidMount中由setTimeout方法中被触发的。

为什么会有这些不同呢?如果你还有印象,在之前的文章中,React对于更新的处理是批量的,也就是说,一系列更新操作会通过某种方式被收集起来,然后统一刷新到页面上。当鼠标事件发生时,组件的最外层会进行处理,然后通过几层包装器的处理后,会开始进行批量更新操作。注意,批量更新操作的前提是确保ReactEventListener处于enabled状态,而且,在前文提到过,在组件的挂载阶段,ReactReconcileTransaction包装器中的一个会先关闭它,以确保整个挂载过程的安全。 至于setTimeout中的调用,则是在将组件放入dirtyComponents列表之前,React会确保事务处于打开状态,在这之后,关闭事务包装器,然后将更新刷新到页面上。

正如你所知,React实现了合成事件,在一下原生事件包裹了一些合成语法糖。在这之后,React会将这些事件处理成常见到样子。看下如下代码里到注释:

为了更好地开发,通过模拟真实游览器事件,我们可以得到更好开发工具整合过程(To help development we can get better dev tool integration by simulating a real browser event’)
var fakeNode = document.createElement('react');

ReactErrorUtils.invokeGuardedCallback = function (name, func, a) {
      var boundFunc = func.bind(null, a);
      var evtType = 'react-' + name;

      fakeNode.addEventListener(evtType, boundFunc, false);

      var evt = document.createEvent('Event');
      evt.initEvent(evtType, false, false);

      fakeNode.dispatchEvent(evt);
      fakeNode.removeEventListener(evtType, boundFunc, false);
};

回到我们的更新流程,我们在这里总结下大概的步骤:
1 调用setState方法
2 如果批量事务未处于开启状态,则打开该事务
3 将受影响组件添加到dirtyComponents列表里
4 通过调用ReactUpdates.flushBatchedUpdates方法关闭事务,也就意味着开始对dirtyComponents列表进行处理。

(未完待续)

相关推荐