了解重排与重绘


Posted in Javascript onMay 29, 2019

前言

浏览器下载完页面中的所有内容:HTML、JavaScript、CSS、图片——之后会解析并生成两个内部数据结构:

  • DOM树:表示页面结构
  • 渲染树:表示DOM节点如何显示

DOM树中的每一个需要显示的节点在渲染树中至少存在一个对应的节点(隐藏的DOM元素在渲染树中没有对应的节点)。渲染树中的节点被称为“帧”或者“盒子”,理解页面元素为一个具有填充(padding)、边距(margin)、边框(border)和位置(position)的盒子。一旦DOM树和渲染树构建完成,浏览器就开始显示(绘制 paint)页面元素

当DOM的改变影响了元素几何属性(例如宽和高)——浏览器就需要重新计算元素的几何特性,同样其他元素的几何属性和位置也会因此受到影响。浏览器会使渲染树中受到影响的部分失效,并重新构造渲染树。这个过程称为重排(reflow)。完成重排后,浏览器会重新绘制受到影响的部分到屏幕中,该过程称为重绘(repaint)

重绘和重排操作都是代价昂贵的操作,它们会导致web应用程序的UI反应迟钝,所以应该尽可能减少这类过程的发生。

引起重排的原因

  • 添加或删除可见的DOM元素
  • 元素位置改变
  • 元素尺寸改变
  • 内容改变(例如:文本改变或图片被另一个不同尺寸的图片替代)
  • 页面渲染初始化
  • 浏览器窗口尺寸改变

渲染树变化的排队与刷新

由于每次重排都会产生计算消耗,大多数浏览器通过队列化修改和批量执行来优化重排过程(将多个重排过程合并成一次)。然而,某些操作会强制刷新队列并要求队列中的重排立即执行(这样会使浏览器的优化策略失效)。

  • 获取布局信息的操作会导致强制刷新队列,使得浏览器不得不执行渲染队列中的“待处理变化”并触发重排以返回正确的值
    • offsetTop,offsetLeft,offsetWidth,offsetHeight
    • scrollTop,scrollLeft,scrollWidth,scrollHeight
    • clientTop,clientLeft,clientWidth,clientHeight
    • getComputedStyle()
  • 优化方法:尽量不要在布局信息改变时查询它,可以在布局信息改变完毕之后再去查询

最小化重绘和重排

重绘和重排的代价非常昂贵,因此一个好的提高程序响应速度的策略就是减少此类操作的发生。为了减少发生次数,应该合并多次对DOM和样式的修改,然后一次处理掉。

合并多次对样式的修改

var el = document.getElementById('myDiv')
el.style.borderLeft = '1px'
el.style.borderRight = '2px'
el.style.padding = '5px'

上面的例子中,存在两个问题:

  • 每个样式属性的改变都会影响元素的集合结构,最糟糕的情况下,会导致浏览器触发三次重排(大部分现代浏览器都为此做了优化,只会触发一次重排)
  • 上面的代码访问了4次DOM

优化方法:使用cssText属性,合并所有的改变然后一次处理

var el = document.getElementById('myDiv')
el.style.cssText +='border-left: 1px; border-right: 2px; padding: 5px;';

批量修改DOM

当需要对DOM元素进行一系列操作时,可以通过以下步骤来减少重绘和重排的次数

1.使元素脱离文档流

2.对其进行一些列操作

3.把元素带回文档中

  • 使元素脱离文档流的方法:
    • 隐藏元素,施加修改,重新显示
    • 使用document fragment在当前DOM之外构建一个子树,再把它拷贝会文档
    • 将元素元素拷贝到另一个脱离文档的节点中,修改副本,完成后再替换原始元素。

缓存布局信息

在上面的介绍中讲到了,浏览器通过队列化修改和批量执行的方式减少重排次数。但是当查询布局信息时(如获取偏移量、滚动位置、计算出的样式值),浏览器为了返回最新值,会刷新队列并应用所有的变更。因此最好的做法应该是尽量减少布局信息的获取次数,获取后把它复制给局部变量,然后再操作局部变量。

IE和:hover

避免在大量元素上使用:hover这种效果。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript对象链式操作代码(jquery)
Jul 04 Javascript
Javascript new关键字的玄机 以及其它
Aug 25 Javascript
javascript获取和判断浏览器窗口、屏幕、网页的高度、宽度等
May 08 Javascript
Javascript中的数组常用方法解析
Jun 17 Javascript
AngularJS在IE下取数据总是缓存问题的解决方法
Aug 05 Javascript
微信小程序-详解数据缓存
Nov 24 Javascript
微信小程序 底部导航栏目开发资料
Dec 05 Javascript
JQuery 选择器、DOM节点操作练习实例
Sep 28 jQuery
JS返回顶部实例代码
Aug 09 Javascript
Vue.js 中 axios 跨域访问错误问题及解决方法
Nov 21 Javascript
微信小程序实现页面浮动导航
Jan 08 Javascript
js前端设计模式优化50%表单校验代码示例
Jun 21 Javascript
小程序如何构建骨架屏
May 29 #Javascript
新手简单了解vue
May 29 #Javascript
详细讲解如何创建, 发布自己的 Vue UI 组件库
May 29 #Javascript
基于JS实现一个随机生成验证码功能
May 29 #Javascript
微信小程序mpvue点击按钮获取button值的方法
May 29 #Javascript
深入了解响应式React Native Echarts组件
May 29 #Javascript
node将geojson转shp返回给前端的实现方法
May 29 #Javascript
You might like
19个超实用的PHP代码片段
2014/03/14 PHP
为你总结一些php信息函数
2015/10/21 PHP
学习php设计模式 php实现建造者模式
2015/12/07 PHP
php调用自己java程序的方法详解
2016/05/13 PHP
php实现与python进行socket通信的方法示例
2017/08/30 PHP
php和nginx交互实例讲解
2019/09/24 PHP
php ActiveMQ的安装与使用方法图文教程
2020/02/23 PHP
SharePoint 客户端对象模型 (一) ECMA Script
2011/05/22 Javascript
用jquery实现的一个超级简单的下拉菜单
2014/05/18 Javascript
关于Javascript加载执行优化的研究报告
2014/12/16 Javascript
jsp 网站引入外部css或者js失效问题解决
2016/10/31 Javascript
基于js 各种排序方法和sort方法的区别(详解)
2018/01/03 Javascript
详解操作虚拟dom模拟react视图渲染
2018/07/25 Javascript
浅谈Vuex注入Vue生命周期的过程
2019/05/20 Javascript
微信小程序跳转到其他网页(外部链接)的实现方法
2019/09/20 Javascript
详解简单易懂的 ES6 Iterators 指南和示例
2019/09/24 Javascript
vue搜索页开发实例代码详解(热门搜索,历史搜索,淘宝接口演示)
2020/04/11 Javascript
[52:15]2014 DOTA2国际邀请赛中国区预选赛5.21 HGT VS LGD-GAMING
2014/05/23 DOTA
[03:08]迎霜节狂欢!2018年迎霜节珍藏Ⅰ一览
2018/12/25 DOTA
深入理解NumPy简明教程---数组2
2016/12/17 Python
Python基于回溯法子集树模板解决取物搭配问题实例
2017/09/02 Python
python爬虫headers设置后无效的解决方法
2017/10/21 Python
Python实现Pig Latin小游戏实例代码
2018/02/02 Python
wx.CheckBox创建复选框控件并响应鼠标点击事件
2018/04/25 Python
python查找特定名称文件并按序号、文件名分行打印输出的方法
2020/04/24 Python
Tensorflow与Keras自适应使用显存方式
2020/06/22 Python
纯css3无js实现的Android Logo(有简单动画)
2013/01/21 HTML / CSS
纯html5+css3下拉导航菜单实现代码
2013/03/18 HTML / CSS
实习教师个人的自我评价
2013/11/08 职场文书
捐款倡议书格式范文
2014/05/14 职场文书
机械工程及其自动化专业求职信
2014/08/08 职场文书
普通党员对照检查材料
2014/09/24 职场文书
HTML通过表单实现酒店筛选功能
2021/05/18 HTML / CSS
python开发的自动化运维工具ansible详解
2021/08/07 Python
vue项目支付功能代码详解
2022/02/18 Vue.js
优化Mysql查询的示例
2022/04/26 MySQL