欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

React 组件渲染和更新的实现代码示例

程序员文章站 2022-03-21 13:57:51
最近一直写react,慢慢就对里面的一些实现很好奇。最好奇的就是自定义标签的实现和this.setstate的实现。这里不分析jsx是如何解析的,所有组件都用es5方式编写...

最近一直写react,慢慢就对里面的一些实现很好奇。最好奇的就是自定义标签的实现和this.setstate的实现。这里不分析jsx是如何解析的,所有组件都用es5方式编写。

组件渲染

渲染时候,我们会调用render方法。类似下面这样:

var sayhi = react.createclass({
 getinitialstate: function() {
  return {verb: 'say:'};
 },
 componentwillmount: function() {
  console.log('i will mount');
 },
 componentdidmount: function() {
  console.log('i have mounted');
 },
 render: function() {
  return react.createelement("div", null,this.state.verb, "hello ", this.props.name);
 }
});

react.render(react.createelement(sayhi, {name: "cynthia"}), document.getelementbyid("container"));

结果:

页面打印:
say: hello cynthia
控制台打印:
i will mount
i have mounted

React 组件渲染和更新的实现代码示例

这是我画的react对象上的一些属性和方法。

当调用render方法时,render会去调用一个map方法,根据传入参数的不同,把被render的对象分为以下三类:
* 文本
* 原生
* 自定义标签

文本

对于文本,react会实例化一个文本节点的对象,并且调用该对象的mount方法。在这个mount方法中,把文本放到一个span中,调用容器组件的innerhtml,进行渲染。

原生标签

对于原生标签,react会实例化一个处理原生标签的对象,并且调用该对象的mount方法。在这个mount方法中,拼接一个字符串,并且不断递归上面的map方法,最后把拼接好的字符串放到容器组件的innerhtml中,进行渲染。

自定义标签

这个应该是大家最好奇的。自定义标签虽然叫标签,其实就是一个类。实例化一个处理自定义标签的对象后,首先react会处理自定义标签的生命周期方法,然后再次递归调用子组件的render方法进而调用map方法,直至把自定义标签分解为前两种标签。

更新

首先,我们统一一下认识。在react里调用this.setstate()会使得组件更新,调用this.state = {}只会更改本组件的状态,但是不会使得组件更新。

如果我要更新一个组件,我会这样写。

var sayhi = react.createclass({
 getinitialstate: function() {
  return {verb: 'say:'};
 },
 componentwillmount: function() {
  console.log('i will mount');
 },
 componentdidmount: function() {
  console.log('i have mounted');
 },
 changeverb: function(){
  this.setstate({verb: 'write:'});
 }
 render: function() {
  return react.createelement("div", this.changeverb.bind(this),this.state.verb, "hello ", this.props.name);
 }
});

react.render(react.createelement(sayhi, {name: "cynthia"}), document.getelementbyid("container"));

执行结果:

页面打印:
say: hello cynthia
点击文本,页面内容更新成:
write: hello cynthia

与更新相关的属性和方法如下:

React 组件渲染和更新的实现代码示例

在调用this.setstate()以后,也是调用了一个map方法,根据传入参数不同,依然把要更新的标签分为文本、原生标签、自定义标签三类。具体处理过程如下。

文本

文本节点处理很简单,判断要更新后的文本与当前文本是否===,不是全等就删除原来文本,插入新文本。

自定义标签

对于自定义标签,首先根据对象的引用、key是否相同,判断是否需要更新。如果需要更新,就继续调用上述map方法进行子组件的更新。又是一个递归。但是注意,这里的map方法和渲染部分的map方法不是一个方法哟。

原生标签

对于原生标签,首先更新组件的属性,然后update子树,用diff算法来比较新的子树与目前标签的子树的不同,形成一个差异树,然后用patch方法,把这个差异树更新到真正的dom树上。

总结

很复杂的过程,让我用流水账写了一遍。没能道出其中精华。以后继续探索,写的详细一些。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。