< 性能提升 Get √ :如何理解 “ 回流 ” 与 “ 重绘 ” ?如何合理的减少其出现呢 ? >

简介: `回流` 和 `重绘` 可以说是每一个web前端开发者经常听到的两个名词,虽然听的多,但是我们真的都理解它们的意思了嘛? 很显然都迷迷糊糊,当然在之前没有去了解的时候,小温也不例外! 之前是介于之前公司要分享,所以当时有做了解,所以这次打算和大伙分享一下!> 🧐 预先剧透一下: " **`回流`** " 和 “ **`重绘`** ” 是阻碍浏览器渲染 及 服务性能的关键因素中 比较重要的两点, 合理的规避它们,能够有效的提高项目的性能!

🗨️ 前言

回流重绘 可以说是每一个web前端开发者经常听到的两个名词,虽然听的多,但是我们真的都理解它们的意思了嘛? 很显然都迷迷糊糊,当然在之前没有去了解的时候,小温也不例外! 之前是介于之前公司要分享,所以当时有做了解,所以这次打算和大伙分享一下!

🧐 预先剧透一下: " 回流 " 和 “ 重绘 ” 是阻碍浏览器渲染 及 服务性能的关键因素中 比较重要的两点, 合理的规避它们,能够有效的提高项目的性能!


👉 一、回流 和 重绘 简介

讲到 回流重绘,自然要从浏览器渲染讲起!为什么呢? 我们要知道,回流 和 重绘不可能单独存在,它们只可能存在于一个完整的步骤或者过程中,而这个过程就是 “ 浏览器渲染的过程 ”。接下来,为了进一步了解 回流 和 重绘,需要先了解浏览器渲染的过程,如下:

image.png

从上面这个图上,我们可以看到,浏览器渲染过程如下:

  • 解析HTML,生成DOM树,解析CSS,生成CSSOM
  • DOM树和CSSOM树结合,生成渲染树(Render Tree)
  • Layout( 回流 ): 根据生成的渲染树,进行回流(Layout),得到节点的几何信息(位置,大小);
  • Painting( 重绘 ):根据渲染树以及回流得到的几何信息,得到节点的绝对像素;
  • Display:将像素发送给GPU,展示在页面上。

在页面初始渲染阶段,回流不可避免的触发,可以理解成页面一开始是空白的元素,后面添加了新的元素使页面布局发生改变

当我们对 DOM 的修改引发了 DOM 几何尺寸的变化(比如修改元素的宽、高或隐藏元素等)时,浏览器需要重新计算元素的几何属性,然后再将计算的结果绘制出来。这叫做 回流

当我们对 DOM 的修改导致了样式的变化(color或background-color),却并未影响其几何属性时,浏览器不需重新计算元素的几何属性、直接为该元素绘制新的样式,这里就仅仅触发了 重绘

根据上面的内容,可以知道:重绘在回流之后,这样一个渲染顺序导致在渲染时,如果出现重绘,它不一定会导致回流,但是如果出现了回流就一定会导致重绘。

👉 二、如何合理减少回流 及 重绘

> 如何触发回流、重绘

要想减少回流和重绘的次数,首先要了解回流和重绘是如何触发的,接下来,详细了解下,它们是如何触发的。

① 回流触发时机
回流这一阶段主要是计算节点的位置和几何信息,那么当页面布局和几何信息发生变化的时候,就需要回流,如下面情况:

  • 添加或删除可见的DOM元素
  • 元素的位置发生变化
  • 元素的尺寸发生变化(包括外边距、内边框、边框大小、高度和宽度等)
  • 内容发生变化,比如文本变化或图片被另一个不同尺寸的图片所替代
  • 页面一开始渲染的时候(这避免不了)
  • 浏览器的窗口尺寸变化(因为回流是根据视口的大小来计算元素的位置和大小的)

还有一些容易被忽略的操作:获取一些特定属性的值

offsetTop、offsetLeft、 offsetWidth、offsetHeight、scrollTop、scrollLeft、scrollWidth、scrollHeight、clientTop、clientLeft、clientWidth、clientHeight

这些属性有一个共性,就是需要通过即时计算得到。因此浏览器为了获取这些值,也会进行回流

除此还包括getComputedStyle 方法,原理是一样的

② 重绘触发时机

触发回流一定会触发重绘

可以把页面理解为一个黑板,黑板上有一朵画好的小花。现在我们要把这朵从左边移到了右边,那我们要先确定好右边的具体位置,画好形状(回流),再画上它原有的颜色(重绘)

除此之外还有一些其他引起重绘行为:

  • 颜色的修改

  • 文本方向的修改

  • 阴影的修改

③ 浏览器优化机制

由于每次重排都会造成额外的计算消耗,因此大多数浏览器都会通过队列化修改并批量执行来优化重排过程。浏览器会将修改操作放入到队列里,直到过了一段时间或者操作达到了一个阈值,才清空队列

当你获取布局信息的操作的时候,会强制队列刷新,包括前面讲到的offsetTop等方法都会返回最新的数据

因此浏览器不得不清空队列,触发回流重绘来返回正确的值。

> 如何减少回流 和 重绘

在上面的内容中,我们了解到了如何触发回流和重绘的场景,为此,只需要按照相反的内容做出预防,即可减少回流和重绘,从而减少性能的损耗了。

下面给出避免回流的经验:

  • 如果想设定元素的样式,通过改变元素的 class 类名 (尽可能在 DOM 树的最里层)
  • 避免设置多项内联样式
  • 应用元素的动画,使用 position 属性的 fixed 值或 absolute 值(如前文示例所提)
  • 避免使用 table 布局,table 中每个元素的大小以及内容的改动,都会导致整个 table 的重新计算
  • 对于那些复杂的动画,对其设置 position: fixed/absolute,尽可能地使元素脱离文档流,从而减少对其他元素的影响
  • 使用css3硬件加速,可以让transform、opacity、filters这些动画不会引起回流重绘
  • 避免使用 CSSJavaScript 表达式

在使用 JavaScript 动态插入多个节点时, 可以使用DocumentFragment. 创建后一次插入. 就能避免多次的渲染,从而导致性能损耗。但有时候,我们会无可避免地进行回流或者重绘,我们可以更好使用它们

例如,多次修改一个把元素布局的时候,我们很可能会如下操作

const el = document.getElementById('el')
for(let i=0;i<10;i++) {
   
   
    el.style.top  = el.offsetTop  + 10 + "px";
    el.style.left = el.offsetLeft + 10 + "px";
}

每次循环都需要获取多次offset属性,比较糟糕,可以使用变量的形式缓存起来,待计算完毕再提交给浏览器发出重计算请求

// 缓存offsetLeft与offsetTop的值
const el = document.getElementById('el') 
let offLeft = el.offsetLeft, offTop = el.offsetTop

// 在JS层面进行计算
for(let i=0;i<10;i++) {
   
   
  offLeft += 10
  offTop  += 10
}

// 一次性将计算结果应用到DOM上
el.style.left = offLeft + "px"
el.style.top = offTop  + "px"

我们还可避免改变样式,使用类名去合并样式

const container = document.getElementById('container')
container.style.width = '100px'
container.style.height = '200px'
container.style.border = '10px solid red'
container.style.color = 'red'

使用类名去合并样式

<style>
    .basic_style {
    
    
        width: 100px;
        height: 200px;
        border: 10px solid red;
        color: red;
    }
</style>
<script>
    const container = document.getElementById('container')
    container.classList.add('basic_style')
</script>

前者每次单独操作,都去触发一次渲染树更改(新浏览器不会),都去触发一次渲染树更改,从而导致相应的回流与重绘过程。合并之后,等于将所有的更改,合并成一次去修改,一次性发出(仅触发一次 回流 和 重绘)。

我们还可以通过通过设置元素属性display: none,将其从页面上去掉,然后再进行后续操作,这些后续操作也不会触发回流与重绘,这个过程称为离线操作。

const container = document.getElementById('container')
container.style.width = '100px'
container.style.height = '200px'
container.style.border = '10px solid red'
container.style.color = 'red'

离线操作后

let container = document.getElementById('container')
container.style.display = 'none'
container.style.width = '100px'
container.style.height = '200px'
container.style.border = '10px solid red'
container.style.color = 'red'
...(省略了许多类似的后续操作)
container.style.display = 'block'

不过离线操作在处理元素较少的情况下,DOM离线操作优越性不太明显,因为离线操作的原理是 将需要处理的元素 “ 拿下 ” 元素,处理完再放上去。这会触发一次性能损耗更大的回流! 但是在需要频繁操作元素的情况下,相比多次触发重绘来说,只触发一两次回流,在全局相比较下,“ 拿下 ”的性能损耗就远远不足为虑了。


📃 参考文献

相关文章
|
9月前
|
缓存 前端开发 JavaScript
|
3月前
|
缓存 JavaScript 前端开发
如何优化事件委托以提高性能?
【10月更文挑战第29天】通过以上优化方法,可以在一定程度上提高事件委托的性能,使其在处理大量DOM元素的事件时更加高效和可靠,为用户提供更流畅的交互体验。在实际应用中,可以根据具体的项目场景和性能瓶颈,有针对性地选择和应用这些优化策略。
|
3月前
|
缓存 JavaScript 前端开发
减少回流和重绘的发生
【10月更文挑战第24天】通过综合运用这些方法,可以有效地减少回流和重绘的发生,提高页面的性能和响应速度,为用户带来更好的体验。
|
3月前
|
存储 缓存 JavaScript
利用缓存布局信息来减少回流和重绘的发生
【10月更文挑战第24天】通过合理利用缓存布局信息,我们可以在一定程度上降低回流和重绘的发生频率,提高页面的性能和用户体验。这是前端性能优化中的一个重要环节,需要我们在实践中不断探索和总结经验,以找到最适合的解决方案。
|
5月前
|
缓存 监控 前端开发
多个异步操作对网页性能的影响及优化建议
多个异步操作会影响网页性能,主要体现在网络请求延迟、资源竞争及浏览器限制等方面,可能导致页面加载缓慢。为优化性能,可采用 `Promise.all()` 并行处理、请求合并、懒加载、缓存利用、CDN 托管、请求优化及性能监控等策略,从而提升用户体验。
|
9月前
|
前端开发 UED 开发者
【专栏】探讨了CSS3动画卡顿的原因,包括复杂动画效果、过多元素参与、低效代码结构和硬件资源限制,并提出优化措施
【4月更文挑战第29天】本文探讨了CSS3动画卡顿的原因,包括复杂动画效果、过多元素参与、低效代码结构和硬件资源限制,并提出优化措施:简化动画路径、控制元素数量、优化代码结构、利用硬件加速及性能监测。通过实际案例展示了优化效果,强调了性能优化对提升用户体验的重要性。在开发中,应持续关注并优化动画性能,以适应网页应用的需求。
473 1
|
9月前
|
缓存 前端开发 JavaScript
优化前端性能:减少页面加载时间的六大技巧
本文将探讨如何通过改善前端性能来减少页面加载时间,从而提升用户体验。我们将介绍六种实用的技巧,包括减少HTTP请求、压缩资源文件、使用CDN加速、延迟加载、缓存策略和代码优化,帮助开发人员有效地提升网页性能。
|
前端开发 JavaScript
前端性能优化—回流与重绘
前端性能优化—回流与重绘
|
算法 测试技术 UED
如何优化模型渲染性能
优化模型渲染性能对于提升用户体验、节约资源消耗、支持复杂场景、扩展到低端硬件、节约能源以及提高开发效率来说都是非常重要的。通过优化性能,可以实现更高质量、更高效的图形渲染。
191 0
|
Serverless
函数计算减少冷启动对性能的影响
函数计算减少冷启动对性能的影响
392 1

热门文章

最新文章