回流与重绘
浏览器的渲染过程
- 解析HTML-->DOM树,解析CSS-->CSSOM树
- DOM树 + CSSOM树 --> Render Tree(渲染树)
回流
(Layout):根据生成的渲染树,回流得到节点的几何信息(位置,尺寸)重绘
(Painting):根据渲染树与回流得到的节点几何信息,得到节点的绝对像素- 将像素发送给
GPU
绘制,然后展示在页面上
生成渲染树过程
- 从DOM根节点开始遍历每个可见的节点
- 不可见节点:
- 不能渲染输出的节点:link,script,meta等等
- 使用css隐藏的节点:display:none;
- 不可见节点:
- 对于每个节点在CSSOM树中找到对应的规则并应用
- 根据每个可见节点与其对应的css样式,组合生产渲染树
回流
计算可见的Dom节点在设备视口的位置和尺寸,这个计算阶段就是回流
为了知道每个可见节点在视口的确切大小和位置,浏览器从渲染树的根节点进行遍历
重绘
经过生成的渲染树和回流阶段,得到了所有可见节点具体的几何信息与样式,然后将渲染树的每个节点转换成屏幕上的实际像素,这个阶段就叫重绘
节点
回流与重绘何时发生?
回流
- 页面第一次加载时
- 浏览器窗口尺寸变化
- 页面布局发生变化
- 添加/删除可见元素
- 节点几何信息发生变化
- 元素的位置发生变化
- 元素的尺寸(外/内边框,外边距,高/宽)
- 内容发生变化
- 文本
- 图片(被另一尺寸图片替换)
重绘
- 回流一定触发重绘
- 可见节点样式发生改变
TIP
回流一定触发重绘,重绘不一定触发回流
如何减少回流与重绘
最小化重绘与重排
- 减少发生次数:合并多次修改
js
const e = document.querySelector('#test')
e.style.color = 'white'
e.style.fontSize = '18px'
const e = document.querySelector('#test')
e.style.color = 'white'
e.style.fontSize = '18px'
- cssTextjs
const e = document.querySelector('#test') e.style.cssText += 'color:white;font-size:18px;'
const e = document.querySelector('#test') e.style.cssText += 'color:white;font-size:18px;'
- classjs
const e = document.querySelector('#test') el.className +='active'
const e = document.querySelector('#test') el.className +='active'
批量修改DOM
需要对dom进行一系列修改时候,减少回流重绘次数方案
- 使元素脱离文档流
- 进行修改
- 将元素放回源文档中
脱离文档流的方式
- 隐藏元素
- 使用
DocumentFragment
构建一个子树,然后拷贝会源文档(document.createDocumentFragment()
) - 拷贝元素到脱离文档流的节点中,修改节点后,在替换原来的节点
- 绝对定位
- 浮动
- 固定定位
其它
- 避免频繁使用 style,而是采用修改class的方式
- resize、scroll 等时间进行防抖或者节流处理
- 添加 will-change: tranform:让渲染引擎为其单独实现一个图层,当这些变换发生时,仅仅只是利用合成线程去处理这些变换,而不牵扯到主线程,大大提高渲染效率
- 动画使用 transform 实现