浅析 网页中的回流(reflow)和重绘(repaints)

浏览器对页面的呈现的处理流程
  1. 文档初次加载时,浏览器引擎会解析HTML文档来构建DOM树
  2. 之后根据DOM元素的几何属性构建一棵用于渲染的树。
  3. 渲染树的每个节点都有大小和边距等属性,类似于盒子模型。
  4. 当渲染树构建完成后,浏览器 就可以将元素放置到正确的位置了,再根据渲染树节点的样式属性绘制出页面。
    这里写图片描述
回流(reflow):

渲染树中的一部分(或全部)因为元素的规模尺寸、布局、隐藏等改变而需要重新构建。

每个页面至少需要一次回流,就是在页面第一次加载的时候。

重绘(repaints):

当渲染树中的一些元素需要更新属性,而这些属性只是影响元素的外观、风格,而不会影响布局,如 background-color。

何时发生(重绘+回流)?

1. 对用户操作(如改变浏览器大小、改变浏览器的字体大小)
2. 增加或者移除样式表
3. 内容变化,比如用户在input框中输入文字
4. 激活 CSS 伪类
5. 操作 class 属性
6. 脚本操作
7. 计算 offsetWidth 和 offsetHeight 属性
8. 隐藏元素(display:none; 回流+重绘 / visiblity:hidden;重绘,不回流)

  • transform 不重绘,不回流
    是因为transform属于合成属性,对合成属性进行transition/animate动画时,将会创建一个合成层。这使得动画元素在一个独立的层中进行渲染。当元素的内容没有发生改变,就没有必要进行重绘。浏览器会通过重新复合来创建动画帧。

  • 回流必将引起重绘,而重绘不一定会引起回流。

var s = document.body.style;
s.padding = "2px"; // 回流+重绘
s.border = "1px solid red"; // 再一次 回流+重绘
s.color = "blue"; // 再一次重绘
s.backgroundColor = "#ccc"; // 再一次 重绘
s.fontSize = "14px"; // 再一次 回流+重绘

document.body.appendChild(document.createTextNode('abc!'));
// 添加node,再一次 回流+重绘
浏览器优化:

从上述代码可以看出,简单的几行代码就引起了6次左右的回流。回流的花销跟 render tree 有多少个节点需要重新构建有关系。如果每句代码都回流重绘的话,浏览器会崩溃的。所以很多浏览器都会优化这些操作。

浏览器会维护一个队列,把所有引起回流重绘的操作放入这个队列,当队列中的操作到达一定数量或者一定时间间隔后,浏览器就会flush队列,进行批处理。这样就会让这些操作变成仅一次的回流重绘,大大减少了浏览器的负担。

但是,有时候我们写的一些代码可能会强制浏览器提前 flush 队列,这样就体现不了优化的好处了。比如执行下面这些操作时,浏览器为了呈现最精准的值,就会 flush 队列。

  • offsetTop,offsetLeft,offsetWidth, offsetHeight
  • scrollTop / Left / Width / Height
  • clientTop / Left / Width / Height
  • width,height
  • 请求了getComputedStyle(),或者 IE的 currentStyle

如何减少回流、重绘?

(在CSS中)

  • 尽可能在 DOM 树的最末端改变 class
  • 避免设置多项内联样式
  • 动画效果应用到 position 属性或 fixed 的元素上
  • 牺牲平滑度换取速度
  • 避免使用 table 布局
  • 避免使用 CSS 的 JavaScript 表达式(仅IE浏览器)

参考1:页面重绘和回流以及优化
参考2:回流与重绘:CSS性能让JavaScript变慢?

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 JavaScript 重绘repaint)和回流(reflow)是与浏览器渲染相关的两个概念。 重绘指的是当元素样式发生改变,但不影响其布局时,浏览器会根据新的样式重新绘制这个元素的过程。重绘并不会引起页面的重新布局,只是改变了元素的外观。 回流指的是当 DOM 结构发生改变或者元素的布局属性发生改变时,浏览器需要重新计算元素的几何属性(位置、大小等),然后重新构建渲染树,这个过程就叫做回流回流会导致页面的重新布局和重绘,比如改变元素的宽高、位置、添加或删除元素等操作都会引起回流。 由于回流涉及到重新计算元素的几何属性以及重新构建渲染树,所以比重绘的代价要高。频繁的回流操作会导致页面性能下降,因此在编写 JavaScript 代码时需要尽量避免频繁的回流操作。 为了减少回流重绘,可以采取以下措施: 1. 使用 CSS3 的 transform 属性来替代 top/left 来进行位置调整,因为 transform 不会引起回流。 2. 对需要多次重绘的元素,使用绝对定位(position: absolute)进行脱离文档流操作,然后再进行修改。 3. 批量修改样式时,可以先将元素的 display 属性设置为 none,然后再进行修改,最后再将 display 属性还原,这样可以减少回流次数。 4. 将需要多次操作的 DOM 元素缓存起来,减少对 DOM 的访问次数。 通过了解重绘回流的概念,并采取相应的优化措施,可以提高页面的性能和用户体验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值