Reflow、Repain的优化

如果你的HTML变得很大很复杂,那么影响你JavaScript性能的可能并不是JavaScript代码的复杂度,而是页面的回流和重绘。

<p><strong>回流</strong>(Reflow)是指布局引擎为frame计算图形的过程。
frame是一个矩形,拥有宽高和相对父容器的偏移。frame用来显示盒模型(content model),
但一个content model可能会显示为多个frame,比如换行的文本每行都会显示为一个frame。</p>

<p><strong>重绘</strong>(Repaint)发生在元素的可见性发生变化时,比如背景色、前景色等。
因此回流必然会引起重绘。</p>

<h1 id="0">HTML 布局</h1>

<p>HTML使用流式布局模型(flow based layout),
这意味着多数情况下一次扫描就可以计算所有的图形显示。
处于流后面的元素一般不会影响前面元素的图形,
所以布局过程可以从左到右、从上到下来进行。</p>

<p>所有的HTML回流都是从根frame开始(HTML标签)的,递归地处理部分或全部子frame。
回流过程中也可能创建新的frame,比如文本发生了换行。
一个frame的回流会导致它的所有父节点以及所有后续元素的回流。</p>

<p>有些HTML回流是立即执行的(immediate to user or script)并且会影响整个frame树,
比如窗口大小变化、更改文档的默认字体;有些HTML回流则是异步的、渐进的(incremental),
比如更多的文档流从网络中到达,这些渐进的回流可以入队列进行批量处理。</p>

<h1 id="1">回流的原因</h1>

<p>浏览器在实现回流时,会递归地处理frame。 每个frame的回流都有一个原因,
这个原因会随着frame逐级向下传递(传递过程中可能会改变)。
回流的原因决定了当前frame的回流行为,有这样5种原因:</p>

<ol>
<li>初始化(Initial)。DOM载入后的第一次回流,将会遍历所有frame。</li>
<li>渐进(Incremental)。当一个frame发生渐进回流时,意味着它前面的元素都没有变,
而是它里面的元素变了。这会引起自底向上的作用。</li>
<li>改变大小(Resize)。元素的容器边界发生变化时,此时元素内部状态没变。
在计算自顶向下的布局约束的同时,可以复用内部状态。</li>
<li>样式改变(StyleChange)。整个frame树都应得到遍历。</li>
<li>Dirty。当一个容器已经缓存了多个子元素的Incremental回流时,该容器出于Dirty的状态。</li>
</ol>

<p>前面四种原因的回流都是在Presentation Shell中立即调用的,
而最后一种回流只有Incremental回流已经到达目标frame时才进行。
(因为这时自底向上的影响才被计算出来,才能决定容器的图形显示)</p>

<p>如果你是Web开发者,可能更关注的是哪些具体原因会引起浏览器的回流,下面罗列一下:</p>

<ol>
<li>调整窗口大小</li>
<li>字体大小</li>
<li>样式表变动</li>
<li>元素内容变化,尤其是输入控件</li>
<li>CSS伪类激活,在用户交互过程中发生</li>
<li>DOM操作,DOM元素增删、修改</li>
<li>CSS3 Animation或Transition</li>
<li>读取元素的某些属性(offsetWidth,offsetTop,offsetHeight,offsetLeft,getComputedStyle 等)</li>
</ol>

<blockquote>
<p>计算这些元素和布局大小时,浏览器会立即Flush渐进回流队列。</p>
</blockquote>

<p>
所以为了避免回流提高页面性能,前端开发需要注意的主要是这两点:<strong>避免大量的DOM操作和布局计算</strong>。</p>

<h1 id="2">最佳实践</h1>

<p>避免布局宽高的计算,意味着尽量使用CSS而不是JS来实现页面的布局效果。
另外一方面,布局宽高的计算结果应当尽量保存避免重复计算。</p>

<p>下面来讨论如何避免大量的DOM操作。请看示例:</p>

  var s = document.body.style; 
s.padding = "2px"; // 回流+重绘
s.border = "1px solid red"; // 再一次 回流+重绘
s.color = "blue"; // 再一次重绘
s.backgroundColor = "#ccc"; // 再一次 重绘
s.fontSize = "14px"; // 再一次 回流+重绘
// 添加node,再一次 回流+重绘
document.body.appendChild(document.createTextNode('abc!'));

可见,通过DOM API进行CSS样式操作会产生大量的回流和重绘。
这一点是可以避免的,比如:预定义CSS类,在JS中进行类的替换(DOM 属性操作)即可。
除此之外,还有很多小技巧可以减少页面回流。下面来总结一下:

  • 避免逐项更改样式。最好一次性更改style属性,或者将样式列表定义为class并一次性更改class属性。
  • 避免循环操作DOM。创建一个documentFragmentdiv,在它上面应用所有DOM操作,最后再把它添加到window.document
  • 也可以在一个display:none的元素上进行操作,最终把它显示出来。因为display:none上的DOM操作不会引发回流和重绘。
  • 避免循环读取offsetLeft等属性。在循环之前把它们存起来。
  • 绝对定位具有复杂动画的元素。绝对定位使它脱离文档刘,否则会引起父元素及后续元素大量的回流
  • 使用CSS3的transition也可以获得不错的性能。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容

  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,727评论 1 92
  • 简介浏览器可以被认为是使用最广泛的软件,本文将介绍浏览器的工 作原理,我们将看到,从你在地址栏输入google.c...
    听风阁阅读 3,270评论 0 7
  • 简介网络浏览器很可能是使用最广的软件。在这篇入门文章中,我将会介绍它们的幕后工作原理。我们会了解到,从您在地址栏输...
    wengjq阅读 1,998评论 2 15
  • 作为一个前端开发,有必要了解从我们在浏览器地址栏输入网址到看到页面这期间浏览器是如何进行工作的,进而了解如何更好的...
    李奕锦liyijin阅读 538评论 0 0
  • 点击题目下方唯美感情学,一键关注本账号 唯美感情学每个女人的内心都有一段不想提及的故事,每个女人都需要一个内心倾述...
    唯美感情学阅读 349评论 0 0