聊一聊React中虚拟DOM
1. 什么是虚拟 DOM
在 React 中实际上是 render 函数中return 的内容会生成 DOM,return 中的内容由两部分组成,一部分是 JSX ,另一部分就是 state 中的数据,所以简单来讲,在 React 中 JSX 结合 state 就生成了 DOM。
现在抛开虚拟 DOM 不谈,如果让我们去实现 React 中当数据发生变化时如何操作 DOM 来实现页面内容的变化,我们会怎样去实现?
第一种方案:
1)JSX + state 生成真实的 DOM,并显示在页面上
2)state 发生变化
3)此时 JSX + state 再次结合生成新的真实的 DOM
4)新的 DOM 直接替换掉原来的 DOM
这样页面会发生变化,但是生成真实的 DOM 和在页面上再重新加载新的 DOM 都比较耗性能。
第二种方案:
1)JSX + state 生成真实的 DOM,并显示在页面上
2)state 发生变化
3)此时 JSX + state 再次结合生成新的真实的 DOM
4)新的 DOM 和原始的 DOM 作对比,找出差异
5)利用找出的差异,替换掉页面上原始 DOM 的相应部分
此时页面也会发生变化,和方案一相比多了对比步骤但是只需要替换掉原始DOM的一部分即可,综合来说,方案二要优于方案一。
第三种方案
1)JSX + state 生成虚拟 DOM(虚拟 DOM 就是一个 JS 对象,用它来描述真实 DOM)
例如下面这段代码:
<div id='abc'>item</div>
注意上面的 div,span 标签时 JSX 语法,并不是真实的 DOM,这里是先生成虚拟 DOM ,然后再下一步的时候才用虚拟 DOM 生成真实的 DOM,由 JSX 到真实的 DOM 中间有一个虚拟 DOM。
JSX -> 虚拟DOM(JS对象) -> 真实DOM
也就是说,JSX 需要先转换为 JS 对象,然后再转换为真实的 DOM。
生成的虚拟 DOM 为
['div',{id: 'abc'}, 'item']
虚拟 DOM 的格式为
['标签名',标签属性对象,子标签]
那么 <div id='abc'>item</div> 是如何转化为 JS 对象的呢?
实际上在 React 中上面这样写就相当于下面这样写:
React.createElement('div', {id: 'abc'}, 'item');
那么实际上就算是没有 JSX 语法通过上面这样写也是可以的,但是会非常不方便。
2)用虚拟 DOM 的结构生成真实的 DOM 显示在页面上。
3)JSX + state 生成新的虚拟 DOM
4)两个虚拟 DOM 进行对比,找出差异
5)根据差异直接修改替换页面上的 DOM
虚拟 DOM 是一个 JS 对象,生成一个虚拟 DOM 比生成一个真实的 DOM 结构要容易省时地多,而且两个虚拟 DOM(JS 对象) 之间的对比也比较简单,所以方案三最佳。
React 中使用的也是第三种方案的思想。
2. 虚拟 DOM 的优点
那么虚拟DOM的优点到底有哪些呢?
1)性能提升
这一点通过上面的比较就可以看得出来
2)使得跨端应用得以实现,例如原生应用。
React Native 能够做原生应用虚拟 DOM 是很重要的一方面,原生应用中是没有 DOM 这个概念的,DOM 是浏览器中存在的,但是有了虚拟 DOM(JS 对象) 之后,在原生应用中就可以将虚拟 DOM(JS 对象) 转换为一些原生应用中能够支持的原生组件在原生应用中显示。
3. 虚拟 DOM 的对比
使用虚拟 DOM 时很重要的一个步骤就是两个虚拟 DOM 之间的比较,那么怎样去进行比较呢?
React 中采用 diff 算法,简单来说主要有以下三个方面:
1)当短时间内连续调用多次 setState 时,React 只会进行一次虚拟 DOM 的比对。
我们知道当 state 或者 props 发生变化时,页面会发生变化,实际上 props 的变化也是因为父组件 state 的变化,所以当页面发生变化时实际上是调用 setState 导致数据发生变化变化时。当短时间内连续调用多次 setState 时,如果每次都进行一次虚拟 DOM 的比对,那么性能会比较低,反之多次调用 setState 只进行一次虚拟 DOM 的比对会提升性能。这也是为什么 setState 要设置成异步的原因,因为如果同步的话当执行完一次 setState 时就会发生一次虚拟 DOM 的比对。(同步是顺序立即执行,异步是当所有的同步程序执行完后再执行)
2)在比较虚拟 DOM 时采用逐层同层比较,当上一层出现差异时,那么下面的各层就不需要再比较了,下面各层的 DOM 都将被新的 DOM 替换。
这样做看起来,复用性不是很好,因为下面各层有可能会有许多相同的 DOM。但是这样做会使得比较算法非常简单,比较的速度非常快。
3)设置 key 值
假设现在有一个数组 [a, b, c] 遍历每一项显示在页面上,现在数组发生变化将第一项 a 删掉,如果没有 key 值,数组 [b, c] 无法和原数组进行比对,例如 b 到底和原数组的哪一个进行比较呢?
但是现在假设有了 key 值,原数组中 a 的 key 值是 a,b 的 key 值是 b,c 的 key 值是 c。删除 a 之后,通过 key 值,b 的 key 值 b 在原数组中找到 b,说明 b 没有发生变化,c 同理也没有发生变化,但是原数组中的 a 在新数组中并没有找到,说明新数组中将 a 删掉了,所以在操作页面时将 a 删掉即可。
这里有一点需要注意的是,key 值一定要选不能变化的,利用数组的索引来做 key 值就不可取。还是以上面为例进行说明。原数组的 a 的 key 值是 0,b 的 key 值是 1,c 的 key 值是 2,删掉 a 后,新数组的 b 的 key 值是 0,c 的 key 值是 1,经过比对原数组的 a 和新数组的 b key 值相同,虚拟 DOM 会认为它们是相同的,没有差异,但是实际上它们是不同的。
聊一聊React中虚拟DOM的更多相关文章
- react中虚拟dom的diff算法
.state 数据 .jsx模板 .生成虚拟dom(虚拟DOM就是一个js对象,用它来描述真实DOM) ['div', {id:'abc'}, ['span', {}, 'hello world']] ...
- react中虚拟DOM的基本概念
react中的核心概念 1.DOM的本质是什么: 浏览器中的概念,用js对象来表示页面上的元素,并提供操作DOM对象的API 2.什么事react中的虚拟DOM:是框架中的概念,是程序员用js对象来模 ...
- react中虚拟DOM
简单来说虚拟DOM就是一个js对象,相对于真实dom来做比较更节约性能,虚拟DOM执行过程如下
- 简单实现react中虚拟DOM渲染
/** * @method createElement * @param type {string} * @param props {Object} * @param children {string ...
- 【React自制全家桶】二、分析React的虚拟DOM和Diff算法
一.React如何更新DOM内容: 1. 获取state 数据 2. 获取JSX模版 3. 通过数据 +模版结合,生成真实的DOM, 来显示,以下行代码为例(简称代码1) <div id= ...
- React的虚拟DOM
ReactJs的一大特点就是引进了虚拟dom(Virtual DOM)的概念.为什么我们需要Virtual DOM,Virtual DOM给我们带来了什么优势. 首先我们要了解一下浏览器的工作流. 当 ...
- React之虚拟DOM中的Diff算法
一.React中的setState ( 异步函数,异步获取数据 ) 若操作的时间间隔短,它可以将多个setState结合成一个setState,减少虚拟DOM的比对次数,提高性能 二.同层虚拟DOM对 ...
- JavaScript是如何工作的:编写自己的Web开发框架 + React及其虚拟DOM原理
这是专门探索 JavaScript 及其所构建的组件的系列文章的第 19 篇. 如果你错过了前面的章节,可以在这里找到它们: JavaScript 是如何工作的:引擎,运行时和调用堆栈的概述! Jav ...
- react 的虚拟dom
前端优化的主要方面就是减少页面的DOM操作,减少重排和重绘,React在这方面做了优化,采用了所谓的虚拟DOM,其实我们平时也会遇到虚拟DOM,只是你没有注意罢了,请听我娓娓道来. 所谓的虚拟DOM ...
随机推荐
- JStorm:任务调度
前一篇文章 JStorm:概念与编程模型 介绍了JStorm的基本概念以及编程模型方面的知识,本篇主要介绍自己对JStorm的任务调度方面的认识,主要从三个方面介绍: 调度角色 调度方法 自定义调度 ...
- php 依赖注入 和 控制反转 php设计模式
https://blog.csdn.net/zyddj123/article/details/82753650 什么是依赖注入?IOC:英文全称:Inversion of Control,中文名称:控 ...
- caffe之数据集介绍
数据集:http://bigdata.51cto.com/art/201702/531276.htm 计算机视觉 MNIST: 最通用的健全检查.25x25 的数据集,中心化,B&W 手写数字 ...
- 两步解决maven plugins 插件下载慢 !下载报红的问题!
两步解决maven plugins 插件下载慢 !下载报红的问题! 1.找到你解压的maven安装路径下的conf 编辑settings 2.添加如下 使用阿里的 <mirror> ...
- 从0开始学正则表达式-基于python
关于正则表达式,当我们了解它就不难,不了解就很难,其实任何事情都是这样,没有人一生下来就啥都会,说白了,每个人都是一个学习了解进步的过程.学习和掌握正则表达式可能并不是太简单,因为它确实是有点像“外星 ...
- Mybatis分页插件的使用流程
如果你也在用Mybatis,建议尝试该分页插件,这一定是最方便使用的分页插件.该插件支持任何复杂的单表.多表分页. 1.引入PageHelper的jar包 在pom.xml中添加如下依赖: 12345 ...
- Job Shop
flow shop: 如果每个作业需要在每个处理机上加工,而且每个作业的工序也相同,即在处理机上加工的顺序相同,则这种多类机的环境称为同顺序作业或流水作业. job shop: 如果每个作业需要在每个 ...
- Archlinux 自动挂载移动硬盘,开机自动启动smb服务
Archlinux + Raspberry 打造NAS: samba篇 树莓派自动挂载硬盘,并开启smb服务. 开机自动挂在移动硬盘ntfs 安装ntfs-3g sudo pacman -S ntfs ...
- 使用powerdesigner进行数据库设计
powerdesigner安装破解文件:链接:https://pan.baidu.com/s/1oKAdUqTKElQ9d86FV-SDTQ 密码:l4y5 基本操作参考:1.PowerDesigne ...
- 高阶函数---swift中的泛型介绍(一步步实现Map函数)
说明 本文内容均出自函数式 Swift一书, 此处整理仅仅是为了自己日后方便查看, 需要深入研究的话, 可以点进去购买, 支持原作者 本书由 王巍–新浪微博大神翻译 OneV's Den 喵神博客 接 ...