关于React中DOM,虚拟DOM及diff算法的理解

前言

初次学习React,最先接触的关键字应该就是响应式UI,虚拟DOM,组件
本文中,我们来讲讲Dom,虚拟Dom,以及diff算法的理解。

DOM

DOM的全称是“Document Object Model”,即文档对象模型。
关于React中DOM,虚拟DOM及diff算法的理解_第1张图片

通过这个截图我们可以发现,其实简单点我们可以将DOM理解为一棵树,一颗倒长的树。也可以说是一颗家谱树,而家谱树本身就是一种模型,其典型用法是表示人类家族谱系。它能很容易表明家族成员之间的关系,把复杂的关系简明地表示出来。

一般的,在传统的Web开发模式中,原生JavaScript是直接对DOM进行更新操作的,这样的做法会非常浪费资源,每操作一次DOM都会对页面进行重新渲染,且新生成一颗DOM树。

对于我们熟悉的jQuery,他是一个JS库。JQuery的Dom操作就是把html里面的元素进行提取和修改然后对文档进行相应的操作。可以很容易的进行样式的设计,内容的提取等等。这样可以很好的操作前端然后获取到你所需要的很多东西。方便了我们的操作和达到我们要求的效果。

而在React中,开发者再也不需要直接与DOM进行沟通。他在开发者与DOM间扮演的一个中间角色,降低了两者之间的开发成本,同时简化了使用的过程。

虚拟DOM

定义:

虚拟DOM其实就是将真实DOM转化为一个抽象的JS对象

背景:

大家都知道,在网页中浏览器资源开销最大便是DOM节点了,DOM很慢并且非常庞大,网页性能问题大多数都是有JavaScript修改DOM所引起的。我们使用Javascript来操纵DOM,操作效率往往很低,由于DOM被表示为树结构,每次DOM中的某些内容都会发生变化,因此对DOM的更改非常快,但更改后的元素,并且它的子项必须经过Reflow / Layout阶段,然后浏览器必须重新绘制更改,这很慢的。因此,回流/重绘的次数越多,您的应用程序就越卡顿。

但是,Javascript运行速度很快,虚拟DOM是放在JS 和 HTML中间的一个层。它可以通过新旧DOM的对比,来获取对比之后的差异对象,然后有针对性的把差异部分真正地渲染到页面上,从而减少实际DOM操作,最终达到性能优化的目的。

树的算法,只会对同层次的两棵树进行比较。遍历时React会逐层次对节点进行比较,只考虑同层次的节点变化,其他层次的节点只进行删除和创建的操作。

此外,虚拟DOM对比与真实DOM没有那么多无用的东西,例如DOM事件,属性等…它只对页面上真正发生变化的内容进行DOM操作,再统一地映射到真实DOM中。所以当浏览器去进行遍历的时候,虚拟DOM的速度是比真实DOM快很多的。

原理:

  • 用JavaScript模拟DOM树,并渲染这个DOM树

  • 比较新老DOM树,得到比较的差异对象

  • 把差异对象应用到渲染的DOM树。
    关于React中DOM,虚拟DOM及diff算法的理解_第2张图片

diff算法

背景

  • 我们已经完成了创建虚拟DOM并将其映射成真实DOM,这样所有的更新都可以先反应到虚拟DOM上,那么是如何反应的呢?这就需要用到Diff算法了。

  • 虚拟DOM性能更快一个重要因素也是使用了diff算法。

  • diff算法的作用是计算出Virtual DOM中真正变化的部分,并只针对该部分进行原生DOM操作,而非重新渲染整个页面。

React中的diff算法

1. 调和

  • 将Virtual DOM树转换成actual DOM树的最少操作的过程称为调和 。
  • diff算法是调和的具体实现。

2. diff策略

React用 三大策略 将O(n^3)复杂度 转化为 O(n)复杂度

  • 策略一(tree diff):
    Web UI中DOM节点跨层级的移动操作特别少,可以忽略不计。
  • 策略二(component diff):
    拥有相同类的两个组件 生成相似的树形结构, 拥有不同类的两个组件 生成不同的树形结构。
  • 策略三(element diff):
    对于同一层级的一组子节点,通过唯一id区分。

参考文档:
https://segmentfault.com/a/1190000016647776
https://www.jianshu.com/p/3ba0822018cf

你可能感兴趣的:(React,笔记,javascript,reactjs,vue.js)