在Web开发中,须要将数据的变化实时反映到UI上,这时就须要对DOM进行操做,可是复杂或频繁的DOM操做一般是性能瓶颈产生的缘由,为此,React引入了虚拟DOM(Virtual DOM)的机制。html
在React中,render执行的结果获得的并非真正的DOM节点,结果仅仅是轻量级的JavaScript对象,咱们称之为virtual DOM。前端
虚拟DOM是React的一大亮点,具备batching(批处理)和高效的Diff算法。这让咱们能够无需担忧性能问题而”毫无顾忌”的随时“刷新”整个页面,由虚拟 DOM来确保只对界面上真正变化的部分进行实际的DOM操做。在实际开发中基本无需关心虚拟DOM是如何运做的,可是理解其运行机制不只有助于更好的理解React组件的生命周期,并且对于进一步优化 React程序也会有很大帮助。vue
若是没有 Virtual DOM,简单来讲就是直接重置 innerHTML。这样操做,在一个大型列表全部数据都变了的状况下,还算是合理,可是,当只有一行数据发生变化时,它也须要重置整个 innerHTML,这时候显然就形成了大量浪费。react
比较innerHTML 和Virtual DOM 的重绘过程以下:git
innerHTML: render html string + 从新建立全部 DOM 元素github
Virtual DOM: render Virtual DOM + diff + 必要的 DOM 更新算法
和 DOM 操做比起来,js 计算是很是便宜的。Virtual DOM render + diff 显然比渲染 html 字符串要慢,可是,它依然是纯 js 层面的计算,比起后面的 DOM 操做来讲,依然便宜了太多。固然,曾有人作过验证说React的性能不如直接操做真实DOM,代码以下:数据库
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
function Raw() {
var data = _buildData(),
html = "";
...
for(var i=0; i<data.length; i++) {
var render = template;
render = render.replace("{{className}}", "");
render = render.replace("{{label}}", data[i].label);
html += render;
}
...
container.innerHTML = html;
...
}
|
该测试用例中虽然构造了一个包含1000个Tag的String,并把它添加到DOM树中,可是只作了一次DOM操做。然而,在实际开发过程当中,这1000个元素更新可能分布在20个逻辑块中,每一个逻辑块中包含50个元素,当页面须要更新时,都会引发DOM树的更新,上述代码就近似变成了以下格式:sublime-text
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
function Raw() {
var data = _buildData(),
html = "";
...
for(var i=0; i<data.length; i++) {
var render = template;
render = render.replace("{{className}}", "");
render = render.replace("{{label}}", data[i].label);
html += render;
if(!(i % 50)) {
container.innerHTML = html;
}
}
...
}
|
这样来看,React的性能就远胜于原生DOM操做了。数组
React.js 相对于直接操做原生DOM有很大的性能优点, 很大程度上都要归功于virtual DOM的batching 和diff。batching把全部的DOM操做搜集起来,一次性提交给真实的DOM。diff算法时间复杂度也从标准的的Diff算法的O(n^3)降到了O(n)。这里留到下一次博客单独讲。
能够看到,Angular 最不效率的地方在于任何小变更都有的和 watcher 数量相关的性能代价。可是!当全部数据都变了的时候,Angular 其实并不吃亏。依赖收集在初始化和数据变化的时候都须要从新收集依赖,这个代价在小量更新的时候几乎能够忽略,但在数据量庞大的时候也会产生必定的消耗。
MVVM 渲染列表的时候,因为每一行都有本身的数据做用域,因此一般都是每一行有一个对应的 ViewModel 实例,或者是一个稍微轻量一些的利用原型继承的 "scope" 对象,但也有必定的代价。因此,MVVM 列表渲染的初始化几乎必定比 React 慢,由于建立 ViewModel / scope 实例比起 Virtual DOM 来讲要昂贵不少。这里全部 MVVM 实现的一个共同问题就是在列表渲染的数据源变更时,尤为是当数据是全新的对象时,如何有效地复用已经建立的 ViewModel 实例和 DOM 元素。假如没有任何复用方面的优化,因为数据是 “全新” 的,MVVM 实际上须要销毁以前的全部实例,从新建立全部实例,最后再进行一次渲染!这就是为何题目里连接的 angular/knockout 实现都相对比较慢。相比之下,React 的变更检查因为是 DOM 结构层面的,即便是全新的数据,只要最后渲染结果没变,那么就不须要作无用功。
Angular 和 Vue 都提供了列表重绘的优化机制,也就是 “提示” 框架如何有效地复用实例和 DOM 元素。好比数据库里的同一个对象,在两次前端 API 调用里面会成为不一样的对象,可是它们依然有同样的 uid。这时候你就能够提示 track by uid 来让 Angular 知道,这两个对象实际上是同一份数据。那么原来这份数据对应的实例和 DOM 元素均可以复用,只须要更新变更了的部分。或者,你也能够直接 track by $index 来进行 “原地复用”:直接根据在数组里的位置进行复用。在题目给出的例子里,若是 angular 实现加上 track by $index 的话,后续重绘是不会比 React 慢多少的。甚至在 dbmonster 测试中,Angular 和 Vue 用了 track by $index 之后都比 React 快: dbmon (注意 Angular 默认版本无优化,优化过的在下面)
在比较性能的时候,要分清楚初始渲染、小量数据更新、大量数据更新这些不一样的场合。Virtual DOM、脏检查 MVVM、数据收集 MVVM 在不一样场合各有不一样的表现和不一样的优化需求。Virtual DOM 为了提高小量数据更新时的性能,也须要针对性的优化,好比 shouldComponentUpdate 或是 immutable data。
(该段落借鉴了知乎的相关回答)
React 历来没有说过 “React 比原生操做 DOM 快”。React给咱们的保证是,在不须要手动优化的状况下,它依然能够给咱们提供过得去的性能。