React中注册机制简析

了解React中的注册机制

在创建React组件方式以及源码解析文章中,有下面一段代码

ReactMount.js源码地址

//下面这个函数实现将ReactElement元素,转化为DOM元素并且插入到对应的Container元素中去;
  _renderNewRootComponent: function (nextElement,container,shouldReuseMarkup,context) {
    //Flag1 下面会有源码解释;
    //instantiateReactComponent(nextElement,false)函数返回一个组件的实例,该函数源码下面会解释;
    var componentInstance = instantiateReactComponent(nextElement,false);

    // The initial render is synchronous but any updates that happen during
    // rendering,in componentWillMount or componentDidMount,will be batched
    // according to the current batching strategy.
    // ReactDefaultInjection.inject();
    //这个函数是真正的将ReactElement元素插入到DOM元素的,会进入到batchedMountComponentIntoNode函数中;
    //这里额外分析React中的注册机制的文章链接 https://github.com/jimwmg/JiM-Blog/tree/master/React
    ReactUpdates.batchedUpdates(batchedMountComponentIntoNode,componentInstance,context);
//这里通过webstorm点击batchedUpdates函数跟踪至源码处:
    var wrapperID = componentInstance._instance.rootID;
    instancesByReactRootID[wrapperID] = componentInstance;

    return componentInstance;    

  }

会跟踪到ReactUpdates.js中

var batchingStrategy = null;

function ensureInjected() {
  !(ReactUpdates.ReactReconcileTransaction && batchingStrategy) ? process.env.NODE_ENV !== 'production' ? invariant(false,'ReactUpdates: must inject a reconcile transaction class and batching strategy') : _prodInvariant('123') : void 0;
}
function batchedUpdates(callback,a,b,c,d,e) {
  ensureInjected();
  return batchingStrategy.batchedUpdates(callback,e);
}
var ReactUpdatesInjection = {
  injectReconcileTransaction: function (ReconcileTransaction) {
    ReactUpdates.ReactReconcileTransaction = ReconcileTransaction;
  },injectBatchingStrategy: function (_batchingStrategy) {
//.....省略一些代码;
    batchingStrategy = _batchingStrategy;
  }
};

var ReactUpdates = {
  ReactReconcileTransaction: null,batchedUpdates: batchedUpdates,enqueueUpdate: enqueueUpdate,flushBatchedUpdates: flushBatchedUpdates,injection: ReactUpdatesInjection,asap: asap
};

module.exports = ReactUpdates;

batchedUpdates函数执行的时候,我们会发现。batchingStrategy 的值是null ; 貌似到这里一切的线索断了,不要着急;

回到ReactDOM.js中,我们看到在最开头有以下一行代码,我们看下它到底做了什么:

ReactDefaultInjection.inject();

ReactDefaultInjection.js中:

var ReactDefaultBatchingStrategy = require('./ReactDefaultBatchingStrategy');
var ReactInjection = require('./ReactInjection');
var alreadyInjected = false;
function inject() {
  if (alreadyInjected) {
    // TODO: This is currently true because these injections are shared between
    // the client and the server package. They should be built independently
    // and not share any injection state. Then this problem will be solved.
    return;
  }
  //.....其余代码不贴了
  alreadyInjected = true;
  //......
  ReactInjection.Updates.injectReconcileTransaction(ReactReconcileTransaction);
  ReactInjection.Updates.injectBatchingStrategy(ReactDefaultBatchingStrategy);
  //这个injectBatchingStrategy函数其实就是ReactUpdates.js中的injectBatchingStrategy;
  //执行这个函数,使得原来在ReactUpdates.js中batchingStrategy = ReactDefaultBatchingStrategy
  //而ReactDefaultBatchingStrategy是一个对象,具体内容就是下面的对象声明;
}

ReactDefaultBatchingStrategy.js中:

var ReactDefaultBatchingStrategy = {
  isBatchingUpdates: false,/**
   * Call the provided function in a context within which calls to `setState`
   * and friends are batched such that components aren't updated unnecessarily.
   */
  //至此,在_renderNewRootComponent中的batchedUpdates函数,就是下面这个函数咯
  batchedUpdates: function (callback,e) {
    var alreadyBatchingUpdates = ReactDefaultBatchingStrategy.isBatchingUpdates;

    ReactDefaultBatchingStrategy.isBatchingUpdates = true;

    // The code is written this way to avoid extra allocations
    if (alreadyBatchingUpdates) {
      return callback(a,e);
    } else {
      return transaction.perform(callback,null,e);
    }
  }
};

module.exports = ReactDefaultBatchingStrategy;

ReactInjection.js中:

var DOMProperty = require('./DOMProperty');
var EventPluginHub = require('./EventPluginHub');
var EventPluginUtils = require('./EventPluginUtils');
var ReactComponentEnvironment = require('./ReactComponentEnvironment');
var ReactEmptyComponent = require('./ReactEmptyComponent');
var ReactBrowserEventEmitter = require('./ReactBrowserEventEmitter');
var ReactHostComponent = require('./ReactHostComponent');
var ReactUpdates = require('./ReactUpdates');

var ReactInjection = {
  Component: ReactComponentEnvironment.injection,DOMProperty: DOMProperty.injection,EmptyComponent: ReactEmptyComponent.injection,EventPluginHub: EventPluginHub.injection,EventPluginUtils: EventPluginUtils.injection,EventEmitter: ReactBrowserEventEmitter.injection,HostComponent: ReactHostComponent.injection,Updates: ReactUpdates.injection
};

module.exports = ReactInjection;

至此,React中的注册机制最基本的实现算是分析完毕,其实React中运用了大量的注册;

如果从整个框架设计的角度来考虑,在最开始的调用ReactDOM.js中进行注册,注册之后,其所引用的其他js中就可以使用了;

React中使用的注册机制,可以很好的节约内容,因为每次注入的对象都是同一个,不会因为每次实例化而生成同样的对象而占据过多的内存;

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


react 中的高阶组件主要是对于 hooks 之前的类组件来说的,如果组件之中有复用的代码,需要重新创建一个父类,父类中存储公共代码,返回子类,同时把公用属性...
我们上一节了解了组件的更新机制,但是只是停留在表层上,例如我们的 setState 函数式同步执行的,我们的事件处理直接绑定在了 dom 元素上,这些都跟 re...
我们上一节了解了 react 的虚拟 dom 的格式,如何把虚拟 dom 转为真实 dom 进行挂载。其实函数是组件和类组件也是在这个基础上包裹了一层,一个是调...
react 本身提供了克隆组件的方法,但是平时开发中可能很少使用,可能是不了解。我公司的项目就没有使用,但是在很多三方库中都有使用。本小节我们来学习下如果使用该...
mobx 是一个简单可扩展的状态管理库,中文官网链接。小编在接触 react 就一直使用 mobx 库,上手简单不复杂。
我们在平常的开发中不可避免的会有很多列表渲染逻辑,在 pc 端可以使用分页进行渲染数限制,在移动端可以使用下拉加载更多。但是对于大量的列表渲染,特别像有实时数据...
本小节开始前,我们先答复下一个同学的问题。上一小节发布后,有小伙伴后台来信问到:‘小编你只讲了类组件中怎么使用 ref,那在函数式组件中怎么使用呢?’。确实我们...
上一小节我们了解了固定高度的滚动列表实现,因为是固定高度所以容器总高度和每个元素的 size、offset 很容易得到,这种场景也适合我们常见的大部分场景,例如...
上一小节我们处理了 setState 的批量更新机制,但是我们有两个遗漏点,一个是源码中的 setState 可以传入函数,同时 setState 可以传入第二...
我们知道 react 进行页面渲染或者刷新的时候,会从根节点到子节点全部执行一遍,即使子组件中没有状态的改变,也会执行。这就造成了性能不必要的浪费。之前我们了解...
在平时工作中的某些场景下,你可能想在整个组件树中传递数据,但却不想手动地通过 props 属性在每一层传递属性,contextAPI 应用而生。
楼主最近入职新单位了,恰好新单位使用的技术栈是 react,因为之前一直进行的是 vue2/vue3 和小程序开发,对于这些技术栈实现机制也有一些了解,最少面试...
我们上一节了了解了函数式组件和类组件的处理方式,本质就是处理基于 babel 处理后的 type 类型,最后还是要处理虚拟 dom。本小节我们学习下组件的更新机...
前面几节我们学习了解了 react 的渲染机制和生命周期,本节我们正式进入基本面试必考的核心地带 -- diff 算法,了解如何优化和复用 dom 操作的,还有...
我们在之前已经学习过 react 生命周期,但是在 16 版本中 will 类的生命周期进行了废除,虽然依然可以用,但是需要加上 UNSAFE 开头,表示是不安...
上一小节我们学习了 react 中类组件的优化方式,对于 hooks 为主流的函数式编程,react 也提供了优化方式 memo 方法,本小节我们来了解下它的用...
开源不易,感谢你的支持,❤ star me if you like concent ^_^
hel-micro,模块联邦sdk化,免构建、热更新、工具链无关的微模块方案 ,欢迎关注与了解
本文主题围绕concent的setup和react的五把钩子来展开,既然提到了setup就离不开composition api这个关键词,准确的说setup是由...
ReactsetState的执行是异步还是同步官方文档是这么说的setState()doesnotalwaysimmediatelyupdatethecomponent.Itmaybatchordefertheupdateuntillater.Thismakesreadingthis.staterightaftercallingsetState()apotentialpitfall.Instead,usecom