在这篇博客中,我们列举一些可以减少 reflow 或者 repaint 的方法。下面先讲述一下这两个概念。
reflow,翻译为重排,意为重新布局。当一个元素的形状位置发生改变时,浏览器会对该元素以及子元素(也可能包括同级元素)进行布局和重绘。称之为重排。
repaint,翻译为重绘。当一个元素的颜色等发生改变时,浏览器会对该元素进行重绘。
为了减少重排和重绘,可以从以下几个方面着手。
批量增删节点
通过elem.innerHTML
批量删除和添加子节点。
通过elem.insertAdjacentHTML
批量添加子节点。
通过创建一个 fragment 来批量添加子节点。
1 | var fragment = document.createDocumentFragment(); |
通过创建一个 range 来批量删除一组连续节点。
1 | function removeChildren(parent, start, end) { |
批量修改节点和样式
通过修改 class 属性代替多次修改 style 属性
不好的做法:
1 | elem.style.width = '200px'; |
较好的做法:
1 | .another-style { |
1 | elem.className = 'another-style'; |
将节点隐藏或者删除后,批量修改,然后再显示或者添加到 DOM 树中。不管修改了多少次,只重新布局两次。
先克隆节点,然后对克隆出来的节点做相应修改,然后再将原节点替换下来。
尽量缩小修改的影响
让对经常变化的节点脱离文档流,比如使用 absolute 或者 fixed 的 position。这样,当该节点重排时不会影响其以后的兄弟节点。
少使用 table 布局,很小的改动会导致整个 table 的重新布局。
访问 DOM 节点的某些属性
比如 offsetTop, offsetLeft, offsetWidth, offsetHeight, scrollTop/Left/Width/Height, clientTop/Left/Width/Height, getComputedStyle()等等,参考When does DOM access slow down rendering?。文章中也提到浏览器越来越聪明,会尽量避免这些不太必要的重排重绘。我暂时还不知道怎么验证是否发生了重排重绘。
留个问题,CSS3 Transition 动画能不能减少重排呢?重绘肯定是少不了的。