1. 什么是虚拟 DOM?
在 React 中,render 执行的结果得到的并不是真正的 DOM 节点,结果仅仅是轻量级的 JavaScript 对象,我们称之为 virtual DOM。
虚拟 DOM 是 React 的一大亮点,具有 batching(批处理) 和高效的 Diff 算法。这让我们可以无需担心性能问题而” 毫无顾忌” 的随时“ 刷新” 整个页面,由虚拟 DOM 来确保只对界面上真正变化的部分进行实际的 DOM 操作。在实际开发中基本无需关心虚拟 DOM 是如何运作的,但是理解其运行机制不仅有助于更好的理解 React 组件的生命周期,而且对于进一步优化 React 程序也会有很大帮助。
2. 虚拟 DOM VS 直接操作原生 DOM?
如果没有 Virtual DOM,简单来说就是直接重置 innerHTML。这样操作,在一个大型列表所有数据都变了的情况下,还算是合理,但是,当只有一行数据发生变化时,它也需要重置整个 innerHTML,这时候显然就造成了大量浪费。
比较 innerHTML 和 Virtual DOM 的重绘过程如下:
innerHTML: render html string + 重新创建所有 DOM 元素
Virtual DOM: render Virtual DOM + diff + 必要的 DOM 更新
和 DOM 操作比起来,js 计算是非常便宜的。Virtual DOM render + diff 显然比渲染 html 字符串要慢,但是,它依然是纯 js 层面的计算,比起后面的 DOM 操作来说,依然便宜了太多。当然,曾有人做过验证说 React 的性能不如直接操作真实 DOM,代码如下:
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 树的更新,上述代码就近似变成了如下格式:
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 操作了。
而且,DOM 完全不属于 Javascript (也不在 Javascript 引擎中存在).。Javascript 其实是一个非常独立的引擎,DOM 其实是浏览器引出的一组让 Javascript 操作 HTML 文档的 API 而已。在即时编译的时代,调用 DOM 的开销是很大的。而 Virtual DOM 的执行完全都在 Javascript 引擎中,完全不会有这个开销。
React.js 相对于直接操作原生 DOM 有很大的性能优势, 很大程度上都要归功于 virtual DOM 的 batching 和 diff。batching 把所有的 DOM 操作搜集起来,一次性提交给真实的 DOM。diff 算法时间复杂度也从标准的的 Diff 算法
3.
简要概括流程有三点:
- 用JavaScript模拟DOM树,并渲染这个DOM树
- 比较新旧DOM树的差异,得到差异对象
- 把差异对象应用到渲染的DOM树上