前言

没有深入研究过, 懂个概念就好, 等性能遇到问题在来看看.

以前写的笔记: 游览器 reflow

参考:

reflow和repaint引发的性能问题

精读《web 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的更多相关文章

  1. 如何优化你的JS脚本来减少reflow/repaint?

    如何优化你的脚本来减少reflow/repaint?1. 避免在document上直接进行频繁的DOM操作,如果确实需要可以采用off-document的方式进行,具体的方法包括但不完全包括以下几种: ...

  2. ♫【网站优化】Reflow / Repaint

    web移动开发最佳实践之js篇 浏览器的回流与重绘 by 张盛志 DOM性能瓶颈与Javascript性能优化 浏览器的渲染原理简介 其中一个跟浏览器有关的原因,那就是浏览器需要花时间.花精力去渲染. ...

  3. 浏览器如何减少 reflow/repaint

    1.不要一条一条地修改 DOM 的样式.与其这样,还不如预先定义好 css 的 class,然后修改 DOM 的 className. 2)把 DOM 离线后修改.如: 使用 documentFrag ...

  4. 浏览器的回流与重绘 (Reflow & Repaint)

    写在前面 在讨论回流与重绘之前,我们要知道: 浏览器使用流式布局模型 (Flow Based Layout). 浏览器会把HTML解析成DOM,把CSS解析成CSSOM,DOM和CSSOM合并就产生了 ...

  5. 介绍回流与重绘(Reflow & Repaint),以及如何进行优化?

    前言 回流与重绘对于前端来说可以说是非常重要的知识点了,我们不仅需要知道什么是回流与重绘,还需要知道如何进行优化.一个页面从加载到完成,首先是构建DOM树,然后根据DOM节点的几何属性形成render ...

  6. Reflow & Repaint

    http://www.phpied.com/rendering-repaint-reflowrelayout-restyle/ http://segmentfault.com/a/1190000002 ...

  7. 浏览器的重绘与回流(Reflow & Repaint)介绍

    重绘 当页面元素样式改变不影响元素在文档流中的位置时(如background-color,border-color,visibility),浏览器只会将新样式赋予元素并进行重新绘制操作. 回流 当改变 ...

  8. 前端性能优化 —— reflow(回流)和repaint(重绘)

    简要:整个在浏览器的渲染过程中(页面初始化,用户行为改变界面样式,动画改变界面样式等)reflow(回流)和repaint(重绘) 会大大影响web性能,尤其是手机页面.因此我们在页面设计的时候要尽量 ...

  9. 前端性能优化 —— reflow(回流/重排)和repaint(重绘)

    简要:整个在浏览器的渲染过程中(页面初始化,用户行为改变界面样式,动画改变界面样式等)reflow(回流)和repaint(重绘) 会大大影响web性能,尤其是手机页面.因此我们在页面设计的时候要尽量 ...

  10. 前端性能优化----reflow(回流)和repaint(重绘)

    什么是reflow和repaint(原文链接:http://www.cnblogs.com/Peng2014/p/4687218.html) reflow:例如某个子元素样式发生改变,直接影响到了其父 ...

随机推荐

  1. webpack4.15.1 学习笔记(二) — 配置及开发环境构建

    目录 基本安装 配置文件 管理资源 管理输出 构建一个开发环境 使用 source map 选择一个开发工具 观察模式 webpack-dev-server webpack-dev-middlewar ...

  2. 手把手帮助你搭建属于自己的个人博客,使用cervel部署,无需后端

    1.项目简介 项目使用了vue+elementUI技术栈,通过读取本地md文件实现博客文章的展示,使用vercel实现自动化部署,纯前端项目,无需后端 第一步:下载源码 仓库地址: github:ht ...

  3. OpenGL之ShadowMap

    流程:先创建一个RenderTexture,然后用灯光的视口渲染. 然后切换到正常相机,进行渲染,使用RenderTexture中的深度或者颜色纹理,然后还原当前顶点在灯光中的深度,两者对比,比缓存中 ...

  4. 智能家居如何把老款定频空调变成智能“变频”空调#米家#智能家居#HA

    背景 最近长沙的天气暴热,室内达到了34-35度,天气预报最高温度上了40度,这么酷热的天气,离开了空调,基本上就是一身汗,全身湿透,特别难受,然后不得不开启家里的一台将近10年的老式定频空调,输入功 ...

  5. python私有变量和方法

    python私有变量和方法 1,私有变量和私有方法无法被继承 2,私有变量和私有方法可以放在普通方法(实例方法)里面被子类继承 class A: def __init__(self, name): s ...

  6. Postman汉化成中文版

    postman安装默认是英文版,为使用方便使用汉化包转成中文版 1.查看本地安装的postman版本:Settings->About 2.根据postman的版本下载对应的汉化包,汉化包网址 3 ...

  7. 基于禅道数据库对bug进行不同维度统计

    工作中经常需要在周报.月报.年报对禅道bug数据进行不同维度统计导出,以下是我常用的统计sql 1.统计2022年每个月bug数(deleted='0'是查询未删除的bug) select DATE_ ...

  8. wordpress站点转移

    title: wordpress站点转移 date: 2024/7/13 11:11:11 tag: linux学习 categories: wordpress建设 description: 搭建后的 ...

  9. 1、Git简介

    1.1.概述 Git 是一个开源免费的分布式版本控制系统,用于快速高效地管理各种小型或大型项目的代码. Git 不仅容易学习.占用空间小,而且性能快如闪电. Git 具有廉价的本地分支.方便的暂存区域 ...

  10. 【Vue】05 Webpack

    Webpack是一个现代JS应用的静态模块打包的工具 学习Webpack需要我们安装NodeJS 配置CNPM & CRM 使用切换镜像的方式配置:[不建议] npm config set r ...