CSS之重绘与回流

重绘(Repaint)

当页面中元素样式的改变并不影响它在文档流中的位置时(例如改变颜色、阴影等),浏览器会进行重绘,即重新绘制元素的外观。

回流(Reflow)

当元素的大小、位置、隐藏等改变时,浏览器需要重新计算元素的几何信息,并重新渲染页面,这个过程称为回流。会影响当前元素、祖先元素和后代元素。

性能影响

  • 回流比重绘的代价要高。回流的过程比重绘复杂,因为它涉及到更多的计算。
  • 回流必将引发重绘,而重绘不一定会引发回流。

触发条件

  • 添加或删除可见的DOM元素
  • 元素尺寸改变
  • 内容变化,例如用户在input框中输入文字
  • 页面渲染初始化
  • 浏览器窗口尺寸改变 ------ 触发重排(回流)。

如何减少重绘和回流

  • 使用transform替代top。
  • 使用visibility替换display: none ------ 因为前者只会触发重绘,后者会触发回流(重排)。
  • 不要使用table布局,因为可能很小的一个小改动会导致整个table的重新布局。
  • 避免在循环中对DOM进行频繁操作,可以使用documentFragment进行批量操作。
  • 避免频繁地读取会引发回流/重绘的属性,如果需要多次使用,可以用变量缓存起来。

代码示例

回流

HTML:

html 复制代码
<div id="container">
    <p>Some text here...</p>
</div>

JavaScript:

javascript 复制代码
let container = document.getElementById('container');

// 这里会触发回流,因为改变了元素的几何属性
container.style.padding = '20px';

// 这又触发了一次回流
container.style.borderWidth = '5px';

为了减少回流,可以合并多次改变样式的操作,使用cssText合并成一次操作:

javascript 复制代码
container.style.cssText = 'padding: 20px; border-width: 5px;';

或者使用CSS类:

css 复制代码
.custom-style {
    padding: 20px;
    border-width: 5px;
}
javascript 复制代码
container.classList.add('custom-style');

重绘

CSS:

css 复制代码
<head>
    <style>
        .box {
            width: 100px;
            height: 100px;
            background-color: blue;
        }
    </style>
</head>
<body>
    <div class="box"></div>
</body>

JavaScript:

javascript 复制代码
document.querySelector('.box').style.backgroundColor = 'red';

这段代码改变了盒子的外观而没有改变它的布局,因此仅仅会导致重绘。

减少重绘和回流

1.使用cssText或者CSS类代替多次DOM操作

2. 使用fragment或clone

当需要对DOM进行大量修改时,最好的做法是使用DocumentFragment或者是clone一个元素,在副本上进行所有操作,然后再把它放回文档中。

javascript 复制代码
let list = document.getElementById('list'),
    frag = document.createDocumentFragment(),
    items = ['Item 1', 'Item 2', 'Item 3'];

// 使用DocumentFragment作操作
items.forEach(item => {
    let li = document.createElement('li');
    li.textContent = item;
    frag.appendChild(li);
});

// 把DocumentFragment一次性添加到DOM中,减少页面回流
list.appendChild(frag);

这样做的好处在于减少了页面的回流次数:不是每添加一个列表项就回流一次,而是所有项都准备好之后才触发一次回流。

相关推荐
UXbot7 分钟前
AI原型设计工具评测:从创意到交互式Demo,5款产品全面解析
前端·ui·设计模式·ai·ai编程·原型模式
落魄江湖行8 分钟前
硅基同事埋的坑,我用2小时才填平:Nuxt 4 路由踩坑:可选参数 [[id]] 与 [id] 的区别
前端
一勺菠萝丶14 分钟前
管理后台使用手册在线预览与首次登录引导弹窗实现
java·前端·数据库
军军君0114 分钟前
Three.js基础功能学习十四:智能黑板实现实例一
前端·javascript·css·typescript·前端框架·threejs·智能黑板
小村儿17 分钟前
连载05-Claude Skill 不是抄模板:真正管用的 Skill,都是从实战里提炼出来的
前端·后端·ai编程
xiaotao13123 分钟前
JS new 操作符完整执行过程
开发语言·前端·javascript·原型模式
robch29 分钟前
python3 -m http.server 8001直接启动web服务类似 nginx
前端·nginx·http
吴声子夜歌36 分钟前
ES6——数组的扩展详解
前端·javascript·es6
guhy fighting44 分钟前
new Map,Array.from,Object.entries的作用以及使用方法
开发语言·前端·javascript
大漠_w3cpluscom1 小时前
CSS 技巧:CSS 单位使用指南
前端