DOM – Browser Reflow & Repaint
前言
没有深入研究过, 懂个概念就好, 等性能遇到问题在来看看.
以前写的笔记: 游览器 reflow
参考:
Reflow & Repaint
repaint 是小的, 性能没有那么伤, 它只是换换颜色之类的.
reflow 是大的, 伤性能, 它会更动到排版布局.
所有插入 dom, 修改 dimension, padding 这类影响 layout 的东西都会导致 reflow.
translate 不会哦, positioin fixed 这种就 ok. 或者说它可以优化. 只是上层渲染, 不会影响整棵树.
when to render ?
当 JS 修改 DOM 时, 游览器并不是马上就渲染的, 它会有一个 timing.
通过 requestanimationframe 可以监听到它渲染了没有. 它不定时的, 依据各种 CPU / GPU 繁忙程度之类的去决定.
所以呢, 可能 JS 修改了多个属性, 但是其实游览器是一次性 reflow, repaint 的.
getBoundingClientRect & window.getComputedStyle
有传言说 getBoundingClientRect 和 getComputedStyle 会导致 reflow.
以前我都不太敢用, 后来才明白它的意思是.
如果 JS 修改了 DOM, 在游览器还没有决定什么时候渲染前, JS 又调用了 getBoundingClientRect.
那么游览器就必须马上去渲染, 这样才能给出准确的 rect. 所以说调用后就会 reflow.
但前提是游览器有需要渲染的东西而它还没有执行渲染. 一旦执行了渲染, 你连续调用 100 次 getBoundingClientRect 它也不会再去 reflow 了.
它怎么可能这么蠢呢. 是我以前蠢.. 哈哈
除了 getBoundingClientRect, getComputedStyle 还有许多操作都会导致提前 reflow 哦,比如 scroll,focus 等等。
Best practice
比如我想 resize table 的每一个 column width。
假如我 for loop 每一个 column,get widest cell then update all column cell to widest。
这样就不好,因为 for loop 的每一次都是一读一写,这样会一直 reflow。
好的做法是 for loop 读取每一个 column 的 widest cell,然后再 for loop 一次 update all column cell to widest。
这样就从 "读写(reflow)读写(reflow)读写(reflow)" 变成了 "读读读写写写(reflow)",最终只 reflow 一次。
具体例子
有一个 h1
<h1>Hello World</h1>
一开始它是 display none
h1 {
display: none;
opacity: 0;
transition: opacity 5s;
} h1.showing {
display: unset;
} h1.shown {
opacity: unset;
}
我们同时给它添加 showing 和 shown class
const h1 = document.querySelector('h1')!;
window.setTimeout(() => {
h1.classList.add('showing');
h1.classList.add('shown');
}, 1000);
它不会有 fade in 的效果,因为游览器同时渲染了 display block 和 opacity 1。(总共渲染 1 次而已)
在两行代码之间加入一些会导致 reflow 的代码
h1.classList.add('showing');
document.body.offsetHeight;
h1.classList.add('showed');
这时就会有 fade in 效果了,因为第一次渲染发生在 document.body.offsetHeight 之前,它只渲染了 display block。
而第二次才渲染 opacity 1。(总共渲染了 2 次)
但凡我们要拿的 information 涉及到布局,十之八九会导致 reflow。
所以一定要切记,如果我们修改了 DOM 又要拿 information 就要很小心,information 最好是可以提前拿啦 (除非你真的是想立刻拿到修改 和 render 后的 information)。
另外一点,调用 getComputedStyle 函数不会导致 reflow 什么的
h1.classList.add('showing');
window.getComputedStyle(h1);
h1.classList.add('showed');
// 只会渲染 1 次而已
但是拿 computedStyle 的属性会导致 reflow
h1.classList.add('showing');
window.getComputedStyle(h1).color;
h1.classList.add('showed');
// 会渲染 2 次
所以同样的,要拿 information 请尽早。
我个人的习惯是把 DOM 操作 wrap 在 requestAnimationFrame 里面,然后里面不做任何的 read information,一定是先把要的 information read 出来,最后才 write。
另外一点,query element (e.g. querySelector, querySelectorAll) 是不会导致立刻 reflow 的
const h1 = document.querySelector('h1')!;
window.setTimeout(() => {
h1.classList.add('showing');
const ww = document.createElement('div');
ww.classList.add('showing');
document.body.appendChild(ww);
const length = document.body.querySelectorAll('.showing').length;
window.setTimeout(() => {
console.log(length);
}, 1000);
h1.classList.add('showed');
}, 1000);
上面只会渲染一次,不会有 fade in 效果。(注:console 会导致 reflow 所以我才用 setTimeout wrap 它起来)
DOM – Browser Reflow & Repaint的更多相关文章
- 如何优化你的JS脚本来减少reflow/repaint?
如何优化你的脚本来减少reflow/repaint?1. 避免在document上直接进行频繁的DOM操作,如果确实需要可以采用off-document的方式进行,具体的方法包括但不完全包括以下几种: ...
- ♫【网站优化】Reflow / Repaint
web移动开发最佳实践之js篇 浏览器的回流与重绘 by 张盛志 DOM性能瓶颈与Javascript性能优化 浏览器的渲染原理简介 其中一个跟浏览器有关的原因,那就是浏览器需要花时间.花精力去渲染. ...
- 浏览器如何减少 reflow/repaint
1.不要一条一条地修改 DOM 的样式.与其这样,还不如预先定义好 css 的 class,然后修改 DOM 的 className. 2)把 DOM 离线后修改.如: 使用 documentFrag ...
- 浏览器的回流与重绘 (Reflow & Repaint)
写在前面 在讨论回流与重绘之前,我们要知道: 浏览器使用流式布局模型 (Flow Based Layout). 浏览器会把HTML解析成DOM,把CSS解析成CSSOM,DOM和CSSOM合并就产生了 ...
- 介绍回流与重绘(Reflow & Repaint),以及如何进行优化?
前言 回流与重绘对于前端来说可以说是非常重要的知识点了,我们不仅需要知道什么是回流与重绘,还需要知道如何进行优化.一个页面从加载到完成,首先是构建DOM树,然后根据DOM节点的几何属性形成render ...
- Reflow & Repaint
http://www.phpied.com/rendering-repaint-reflowrelayout-restyle/ http://segmentfault.com/a/1190000002 ...
- 浏览器的重绘与回流(Reflow & Repaint)介绍
重绘 当页面元素样式改变不影响元素在文档流中的位置时(如background-color,border-color,visibility),浏览器只会将新样式赋予元素并进行重新绘制操作. 回流 当改变 ...
- 前端性能优化 —— reflow(回流)和repaint(重绘)
简要:整个在浏览器的渲染过程中(页面初始化,用户行为改变界面样式,动画改变界面样式等)reflow(回流)和repaint(重绘) 会大大影响web性能,尤其是手机页面.因此我们在页面设计的时候要尽量 ...
- 前端性能优化 —— reflow(回流/重排)和repaint(重绘)
简要:整个在浏览器的渲染过程中(页面初始化,用户行为改变界面样式,动画改变界面样式等)reflow(回流)和repaint(重绘) 会大大影响web性能,尤其是手机页面.因此我们在页面设计的时候要尽量 ...
- 前端性能优化----reflow(回流)和repaint(重绘)
什么是reflow和repaint(原文链接:http://www.cnblogs.com/Peng2014/p/4687218.html) reflow:例如某个子元素样式发生改变,直接影响到了其父 ...
随机推荐
- 解决方案 | pywintypes.com_error: (-2147221005, '无效的类字符串', None, None) --Python连接CAD报错真正解决思路!
1 背景 import pythoncom import win32com.client import math wincad = win32com.client.Dispatch("Aut ...
- leetcode中等(字符串):[3, 6, 8, 49, 179, 299, 524, 539, 609, 648]
目录 3. 无重复字符的最长子串 6. Z 字形变换 8. 字符串转换整数 (atoi) 49. 字母异位词分组 179. 最大数 299. 猜数字游戏 524. 通过删除字母匹配到字典里最长单词 5 ...
- oeasy教您玩转vim - 21 - 状态横条
状态横条 回忆上节课内容 我们上次研究了标尺 标尺 开启 se ru 关闭 se noru 行号 开启 se nu 关闭 se nonu 命令位置 开启 se showcmd 关闭 se noshow ...
- Prometheus 基于Python Django实现Prometheus Exporter
基于Python Django实现Prometheus Exporter 需求描述 运行监控需求,需要采集Nginx 每个URL请求的相关信息,涉及两个指标:一分钟内平均响应时间,调用次数,并且为每个 ...
- vscode 常用设置
vscode 常用设置 by:授客 QQ:1033553122 版本 vscode Version: 1.33.1 1. 自动保存文件设置 文件编辑一秒钟过后自动保存 2. 黏贴后 ...
- Goutte爬虫
安装 composer require fabpot/goutte:4.0
- 不是,大哥,咱这小门小户的,别搞我CDN流量啊
分享是最有效的学习方式. 博客:https://blog.ktdaddy.com/ 最近遇上事儿了,老猫的小小博客网站[程序员老猫的博客]被人盗刷CDN流量了.开始的时候收到了欠费的短信也没有怎么去重 ...
- P9058 [Ynoi2004] rpmtdq 与 P9678 [ICPC2022 Jinan R] Tree Distance
思路: 注意到点对数量有 \(N^2\) 个,考虑丢掉一些无用的点对. 对于点对 \((x_1,y_1),(x_2,y_2)\),满足 \(x_1 \le x_2 < y_2 \le y_1\) ...
- 第一讲:日志系统:一条sql更新语句是如何执行的?
目录 第一讲:日志系统:一条sql更新语句是如何执行的? 回顾 抛出问题: 提出思路: 重要的日志模块:redo log 重要的日志模块:binlog 为什么会有两份日志呢? 这两种日志有以下三点不同 ...
- PyTorch显存机制分析
参考: ======================================================= 在pytorch中有几个关于显存的关键词: 在pytorch中显存为缓存和变量分 ...