当前位置: 首页 > news >正文

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

前言

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

DOM

DOM的全称是“Document Object Model”,即文档对象模型。
在这里插入图片描述

通过这个截图我们可以发现,其实简单点我们可以将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树。
    在这里插入图片描述

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

相关文章:

  • 类型签名在Javascript中的探索
  • 快速入门,理解,使用 axios请求
  • 第二章 存储,2.1 永不停止的脚步——数据库优化之路(作者:佳毅)
  • CSS实现元素水平垂直居中的几种方式
  • 年末购机推荐,首选OPPO这两款中端旗舰王者
  • js中深浅拷贝的实现方式(含图解原理)
  • Java刷题知识点之File对象常用功能:获取文件名称、获取文件路径、获取文件大小、获取文件修改时间、创建与删除、判断、重命名、查看系统根目录、容量获取、获取某个目录下内容、过滤器...
  • 查看httpd状态
  • js中如何判断引用值为数组(几种不同方式的详解)
  • 项目代码重用
  • js中数组去重的几种实现方式(区别)
  • java创建文件和目录
  • JS For循环中嵌套setTimeout()方法的理解
  • ubuntu下zabbix服务器监控工具部署
  • 前端性能优化-图片
  • 分享一款快速APP功能测试工具
  • __proto__ 和 prototype的关系
  • 【mysql】环境安装、服务启动、密码设置
  • 【刷算法】从上往下打印二叉树
  • Druid 在有赞的实践
  • electron原来这么简单----打包你的react、VUE桌面应用程序
  • input实现文字超出省略号功能
  • Java|序列化异常StreamCorruptedException的解决方法
  • JavaScript 事件——“事件类型”中“HTML5事件”的注意要点
  • Vultr 教程目录
  • 第三十一到第三十三天:我是精明的小卖家(一)
  • 分布式任务队列Celery
  • 突破自己的技术思维
  • 用Canvas画一棵二叉树
  • - 语言经验 - 《c++的高性能内存管理库tcmalloc和jemalloc》
  • ​力扣解法汇总946-验证栈序列
  • # 安徽锐锋科技IDMS系统简介
  • # 再次尝试 连接失败_无线WiFi无法连接到网络怎么办【解决方法】
  • (1)Map集合 (2)异常机制 (3)File类 (4)I/O流
  • (Mac上)使用Python进行matplotlib 画图时,中文显示不出来
  • (二)七种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (附源码)spring boot校园拼车微信小程序 毕业设计 091617
  • (四)Linux Shell编程——输入输出重定向
  • (一)Mocha源码阅读: 项目结构及命令行启动
  • (一)Neo4j下载安装以及初次使用
  • .NET / MSBuild 扩展编译时什么时候用 BeforeTargets / AfterTargets 什么时候用 DependsOnTargets?
  • .net 无限分类
  • .Net各种迷惑命名解释
  • .NET建议使用的大小写命名原则
  • .net利用SQLBulkCopy进行数据库之间的大批量数据传递
  • .Net下的签名与混淆
  • @EnableWebMvc介绍和使用详细demo
  • @hook扩展分析
  • [20171101]rman to destination.txt
  • [bzoj1912]异象石(set)
  • [dart学习]第四篇:函数
  • [FT]chatglm2微调
  • [HTML]HTML5实现可编辑表格
  • [LeetCode] Wildcard Matching
  • [LeetCode刷题笔记]1 - 两数之和(哈希表)