了解重排与重绘


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作一个通用向导说明
Aug 30 Javascript
IE下通过a实现location.href 获取referer的值
Sep 04 Javascript
JavaScript中的toDateString()方法使用详解
Jun 12 Javascript
为何JS操作的href都是javascript:void(0);呢
Nov 12 Javascript
jQuery select自动选中功能实现方法分析
Nov 28 Javascript
jQuery插件HighCharts绘制2D柱状图、折线图和饼图的组合图效果示例【附demo源码下载】
Mar 09 Javascript
基于Bootstrap的标签页组件及bootstrap-tab使用说明
Jul 25 Javascript
JS实现评价的星星功能
Aug 20 Javascript
从零到一详聊创建Vue工程及遇到的常见问题
Apr 25 Javascript
vue.js 2.0实现简单分页效果
Jul 29 Javascript
超详细的5个Shell脚本实例分享(值得收藏)
Aug 15 Javascript
vue route新窗口跳转页面并且携带与接收参数
Apr 10 Vue.js
小程序如何构建骨架屏
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
php 运行效率总结(提示程序速度)
2009/11/26 PHP
PHP5权威编程阅读学习笔记 附电子书下载
2012/07/05 PHP
JQuery 自定义CircleAnimation,Animate方法学习笔记
2011/07/10 Javascript
JS对select控件option选项的增删改查示例代码
2013/10/21 Javascript
JQuery中$.each 和$(selector).each()的区别详解
2015/03/13 Javascript
Javascript实现网络监测的方法
2015/07/31 Javascript
Angular路由简单学习
2016/12/26 Javascript
javascript实现动态显示颜色块的报表效果
2017/04/10 Javascript
jquery实现用户登陆界面(示例讲解)
2017/09/06 jQuery
javaScript字符串工具类StringUtils详解
2017/12/08 Javascript
vue2.0模拟锚点的实例
2018/03/14 Javascript
微信小程序实现左右联动的实战记录
2018/07/05 Javascript
微信小程序日历组件使用方法详解
2018/12/29 Javascript
简单学习5种处理Vue.js异常的方法
2019/06/17 Javascript
Vue.js递归组件实现组织架构树和选人功能
2019/07/04 Javascript
ES6 新增的创建数组的方法(小结)
2019/08/01 Javascript
vue图片加载失败时用默认图片替换的方法
2019/08/29 Javascript
[03:17]DOTA2-DPC中国联赛1月29日Recap集锦
2021/03/11 DOTA
400多行Python代码实现了一个FTP服务器
2012/05/10 Python
Python创建日历实例
2014/08/21 Python
Python实现树莓派WiFi断线自动重连的实例代码
2017/03/16 Python
Python scikit-learn 做线性回归的示例代码
2017/11/01 Python
通过python+selenium3实现浏览器刷简书文章阅读量
2017/12/26 Python
python使用多进程的实例详解
2018/09/19 Python
windows下 兼容Python2和Python3的解决方法
2018/12/05 Python
Django中Q查询及Q()对象 F查询及F()对象用法
2020/07/09 Python
基于python爬取链家二手房信息代码示例
2020/10/21 Python
HTML5无刷新改变当前url的代码
2017/03/15 HTML / CSS
意大利宠物用品购物网站:Bauzaar
2018/09/15 全球购物
Hobbs官方网站:英国奢华女性时尚服装
2020/02/22 全球购物
房地产销售经理岗位职责
2014/01/01 职场文书
社会实践先进工作者事迹材料
2014/05/06 职场文书
学期个人工作总结
2015/02/13 职场文书
2016教师政治学习心得体会
2016/01/23 职场文书
nginx常用命令放入shell脚本详解
2021/03/31 Servers
Python 机器学习工具包SKlearn的安装与使用
2021/05/14 Python