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


    HTML加载时发生了什么

      在页面加载时,浏览器把获取到的HTML代码解析成1个DOM树,DOM树里包含了所有HTML标签,包括display:none隐藏,还有用JS动态添加的元素等。
      浏览器把所有样式(用户定义的CSS和用户代理)解析成样式结构体
      DOM Tree 和样式结构体组合后构建render tree, render tree类似于DOM tree,但区别很大,因为render tree能识别样式,render tree中每个NODE都有自己的style,而且render tree不包含隐藏的节点(比如display:none的节点,还有head节点),因为这些节点不会用于呈现,而且不会影响呈现的,所以就不会包含到 render tree中。我自己简单的理解就是DOM Tree和我们写的CSS结合在一起之后,渲染出了render tree.

    回流

      当render tree中的一部分或全部因为元素的规模尺寸、布局、隐藏等改变时,浏览器重新渲染部分DOM或全部DOM的过程

      称为回流。

       每个页面至少需要一次回流,就是在页面第一次加载的时候,这时候是一定会发生回流的,因为要构建render tree。在回流的时候,浏览器会使渲染树中受到影响的部分失效,并重新构造这部分渲染树,完成回流后,浏览器会重新绘制受影响的部分到屏幕中,该过程成为重绘。

    重绘

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

    下面情况会导致reflow发生:

    1:改变窗口大小
    
    2:改变文字大小
    
    3:内容的改变,如用户在输入框中敲字
    
    4:激活伪类,如:hover
    
    5:操作class属性
    
    6:脚本操作DOM
    
    7:计算offsetWidth和offsetHeight
    
    8:设置style属性

    如何进行优化呢?

      1.浏览器的优化机制:

        现代的浏览器都是很聪明的,由于每次重排都会造成额外的计算消耗,因此大多数浏览器都会通过队列化修改并批量执行来优化重排过程。浏览器会将修改操作放入到队列里,直到过了一段时间或者操作达到了一个阈值,才清空队列。但是!当你获取布局信息的操作的时候,会强制队列刷新,比如当你访问以下属性或者使用以下方法:

    offsetTop、offsetLeft、offsetWidth、offsetHeight
    scrollTop、scrollLeft、scrollWidth、scrollHeight
    clientTop、clientLeft、clientWidth、clientHeight
    getComputedStyle()
    getBoundingClientRect

    以上属性和方法都需要返回最新的布局信息,因此浏览器不得不清空队列,触发回流重绘来返回正确的值。因此,我们在修改样式的时候,最好避免使用上面列出的属性,他们都会刷新渲染队列。

    如果我们要经常去获取和操作这些值,则可以先将这些值缓存起来例如:

      

    var windowHeight = window.innerHeight;//reflow
    
    for(i=0;i<10;i++){
    
      $body.height(windowHeight++);
      一系列关于windowHeight的操作....... }

      

      2.自己的优化

      靠浏览器不如靠自己,我们可以改变一些写法减少回流和重绘:

     

      1:不要通过父级来改变子元素样式,最好直接改变子元素样式,改变子元素样式尽可能不要影响父元素和兄弟元素的大小和尺寸。

       2:尽量通过class来设计元素样式,切忌用style。

       3.对于复杂动画效果,使用绝对定位让其脱离文档流,否则会引起父元素及后续元素大量的回流。

       4:不要用tables布局的另一个原因就是tables中某个元素一旦触发reflow就会导致table里所有的其它元素reflow。在适合用table的场合,可以设置table-layout为auto或fixed.

       5.避免设置多层内联样式.

      6.使用修改样式cssText.

      cssText 本质是什么?

      cssText 的本质就是设置 HTML 元素的 style 属性值。

      cssText 怎么用?

    document.getElementById("d1").style.cssText = "color:red; font-size:13px;";

      cssText 返回值是什么?

    在某些浏览器中(比如 Chrome),你给他赋什么值,它就返回什么值。在 IE 中则比较痛苦,它会格式化输出、会把属性大写、会改变属性顺序、会去掉最后一个分号,比如:

     
    1
    2
    document.getElementById("d1").style.cssText = "color:red; font-size:13px;";
    alert(document.getElementById("d1").style.cssText);

    在 IE 中值为:FONT-SIZE: 13px; COLOR: red

     

      js中有一个cssText的方法:

     

      语法为:

     

      obj.style.cssText=”样式”;

     

      element.style.cssText=”20px;height:20px;border:solid 1px red;”

     

      比如改变样式的时候,不去改变他们每个的样式,而是直接改变className 就要用到cssText 但是要注意有一个问题,会把原有的cssText清掉,比如原来的style中有’display:none;’,那么执行完上面的JS后,display就被删掉了。
    为了解决这个问题,可以采用cssText累加的方法
    Element.style.cssText += ‘100px;height:100px;top:100px;left:100px;’

     

    但是IE不支持累加,前面添一个分号可以解决。
    Element.style.cssText += ‘;100px;height:100px;top:100px;left:100px;’


    前端怎么学的会~
  • 相关阅读:
    my34_脚本冥等添加自动任务-mysql监控部署
    zabbix4.2 安装
    面试题12:字符串无重复子串的最大长度
    面试题11:二叉树的非递归前、中、后、层级遍历
    面试题10:二叉树的最大路径和
    面试题9:数组堆化、堆的插入、堆的删除、堆排序
    面试题8:无序数组的最大差值
    面试题7:判断链表是否有环,返回环的入口点
    面试题6:二叉树最近公共节点(LCA)《leetcode236》
    面试题6:二叉树转单链表
  • 原文地址:https://www.cnblogs.com/hope192168/p/12000752.html
Copyright © 2020-2023  润新知