重排与重绘

重排(Reflow)与重绘(Repaint)

改布局触发重排(必连带重绘);只改外观只触发重绘;读布局属性会强制立即重排。


一、对比速查

重排(Reflow) 重绘(Repaint)
定义 几何布局变了,浏览器重新算位置和大小 外观变了、布局不变,浏览器只重画像素
开销 🔴 大 🟡 小
触发属性 width height margin padding top left border font-size line-height color background border-color opacity visibility box-shadow outline
关系 重排 一定连带 重绘 重绘 不一定 触发重排

二、触发时机详解

触发重排的 3 类操作

① DOM 结构变化

增删可见节点、改文本内容、改层级(z-index 不影响布局不算)

② 几何属性变化

css 复制代码
width / height    margin / padding
top / left        border / border-width
font-size         line-height
display           position

③ 读取布局属性(强制同步重排 --- 最坑)

读以下属性时浏览器必须清空队列、立即执行重排:

属性族 具体属性
offset offsetTop offsetLeft offsetWidth offsetHeight
scroll scrollTop scrollLeft scrollWidth scrollHeight
client clientTop clientLeft clientWidth clientHeight
方法 getBoundingClientRect() getComputedStyle()

④ 窗口 resize

浏览器窗口大小改变,全局重排。

触发重绘的操作

所有只改变外观、不影响布局的 CSS 属性:

css 复制代码
color / background* / border-color / outline
opacity / visibility
box-shadow / text-shadow

三、浏览器优化:队列化机制

正常改样式时,浏览器会把多次 DOM 操作放进队列,攒一批再一起重排/重绘,减少次数。

但一旦读取布局属性(offset/scroll/client 族),浏览器必须立刻清空队列、立即重排才能给出准确值。这就是「强制同步布局」(Forced Synchronous Layout)------打断优化的主要原因。


四、代码示例

js 复制代码
// ✅ 只触发重绘
box.style.backgroundColor = 'blue';
box.style.opacity = 0.5;

// ⚠️ 触发重排 + 重绘
box.style.width = '200px';
box.style.left = '10px';

// ❌ 读布局 → 强制立即重排(典型坑)
console.log(box.offsetWidth);

常见反模式 vs 优化

js 复制代码
// ❌ 读写交错的强制同步布局
box.style.width = '200px';       // 写入队列
const w = box.offsetWidth;       // 强制重排!
box.style.height = w + 'px';     // 再次写入

// ✅ 先批量读,再批量写
const w = box.offsetWidth;       // 读
const h = box.offsetHeight;      // 读
box.style.width = '200px';       // 写
box.style.height = h + 'px';     // 写(一次重排搞定)

五、一句话速记

重排 = 布局动了(位置/大小)→ 开销大;重绘 = 样子变了(颜色/透明)→ 开销小;读 offset/scroll/client 族属性 → 强制立即重排。

相关推荐
LinXunFeng7 小时前
Obsidian - 使用 Share Note 分享笔记并自部署
前端·笔记·github
乘风gg11 小时前
为什么AI 时代来临,大部分人吃不到红利
前端·ai编程·claude
恋猫de小郭11 小时前
Android 限制侧载新进展,谷歌联合国内厂商推验证计划
android·前端·flutter
IT_陈寒11 小时前
Redis内存爆了,原来我漏掉了这个致命配置
前端·人工智能·后端
恋猫de小郭11 小时前
解读 Android 17 全新内存限制,有没有“豁免”后门?
android·前端·flutter
Hyyy13 小时前
理解LLM的基本工作原理:预训练、微调、推理的区别
前端
Gatlin13 小时前
前端逆向与反逆向:一场猫鼠游戏的底层逻辑与实战
前端
代码煮茶13 小时前
React 组件封装方法论 —— 以 Todo App 为例
javascript·react.js
Pedantic13 小时前
本地通知(Local Notifications)学习笔记
前端
任沫13 小时前
Agent之Function Call
javascript·人工智能·go