createDocumentFragment
如果要在一个节点上一次性插入多个元素怎么办,比如说一次插入 10000 个节点?
最简单粗暴的方式就是:
var parent = document.getElementById(`'parent'`);for`(`var i = 0; i < 10000; i++) {
var child = document.createElement(`'div'`);
var text = document.createTextNode(`'' + i);`
child.appendChild(text);
parent.appendChild(child);
}
不过众所周知的原因,对 DOM 反复操作会导致页面重绘、回流,效率非常低,而且页面可能会被卡死,这段代码基本是没人用的。
如果分段来进行 DOM 操作呢,这样就能避免卡死页面了,js 忍者秘籍里面提到过可以用 setTimeout 来改进:
var i = 0, max = 10000;setTimeout(`function addNodes() {`
for`(`var step = i + 500; i < step; i++) {
var child = document.createElement(`'div'`);
child.appendChild(document.createTextNode(`'' + i));`
div.appendChild(child);
}
if`(i < max) {`
setTimeout(addNodes, 0);
}
}, 0);
当然,更多能想到的方式应该是,在内存中直接操作节点,所有节点都凑在一起之后再跟 DOM 树进行交互,把所有节点都串在一个 div 上,然后再把 div 挂到 DOM 树上:
var parent = document.getElementById(`'parent'`);var div = document.createElement(`'div'`);
for`(`var i = 0; i < 10000; i++) {
var child = document.createElement(`'div'`);
var text = document.createTextNode(`'' + i);`
child.appendChild(text);
div.appendChild(child);
}
parent.appendChild(div);
如上,只跟 DOM 树交互一次,性能方面肯定是大有改善的,不过额外插入了一个 div,如果说不是跟div之类的节点进行交互呢,比如在 table 中插入 th、td?
这时候,createDocumentFragment 就该出马了,翻译过来叫“文档片段”,按MDN的描述:
DocumentFragments 是一些 DOM 节点。它们不是 DOM 树的一部分。通常的使用场景是创建一个文档片段,然后将创建的 DOM 元素插入到文档片段中,最后把文档片段插入到 DOM 树中。在 DOM 树中,文档片段会被替换为它所有的子元素。
因为文档片段存在与内存中,并不在 DOM 树中,所以将子元素插入到文档片段时不会引起页面回流(对元素位置和几何上的计算)。因此,使用文档片段 document fragments 通常会起到优化性能的作用。
简单来说,就是上面一个例子的不需要 div 中转版本,插入的时候,直接用其子元素替换其本身,非常完美。
虽然说,“好用的都不通用”(特别是针对某公司浏览器),不过这个好用的东西,甚至连 IE6 都支持。










