zoukankan      html  css  js  c++  java
  • Dom 优化

    核心问题

        当解析的html文件很大时,生成DOM树占用内存较大,同时遍历(不更新)元素耗时也更长。但这都不是重点,DOM的核心问题是:DOM修改导致的页面重绘、重新排版!重新排版是用户阻塞的操作,同时,如果频繁重排,CPU使用率也会猛涨!

        DOM操作会导致一系列的重绘(repaint)、重新排版(reflow)操作。为了确保执行结果的准确性,所有的修改操作是按顺序同步执行的。大部分浏览器都不会在JavaScript的执行过程中更新DOM。相应的,这些浏览器将对对 DOM的操作放进一个队列,并在JavaScript脚本执行完毕以后按顺序一次执行完毕。也就是说,在JavaScript执行的过程,直到发生重新排版,用户一直被阻塞。

        一般的浏览器中(不含IE),repaint的速度远快于reflow,所以避免reflow更重要

        导致repaint、reflow的操作

        * DOM元素的添加、修改(内容)、删除( Reflow + Repaint)

        * 仅修改DOM元素的字体颜色(只有Repaint,因为不需要调整布局)

        * 应用新的样式或者修改任何影响元素外观的属性

        * Resize浏览器窗口、滚动页面

        * 读取元素的某些属性(offsetLeft、offsetTop、offsetHeight、offsetWidth、scrollTop/Left/Width/Height、clientTop/Left/Width/Height、getComputedStyle()、currentStyle(in IE))

    Javascript的应对方案

    【1】在DOM外,执行尽量多的变更操作。Demo

    // 不好的做法
    for (var i=0; i < items.length; i++){
        var item = document.createElement("li");
        item.appendChild(document.createTextNode("Option " + i);
        list.appendChild(item);
    }   
     
     
    // 更好的做法
    // 使用容器存放临时变更, 最后再一次性更新DOM
    var fragment = document.createDocumentFragment();
    for (var i=0; i < items.length; i++){
        var item = document.createElement("li");
        item.appendChild(document.createTextNode("Option " + i);
        fragment.appendChild(item);
    }
    list.appendChild(fragment);

    【2】操作DOM前,先把DOM节点删除或隐藏,因为隐藏的节点不会触发重排。Demo如下:

    复制代码
    list.style.display = "none";  
    for (var i=0; i < items.length; i++){  
        var item = document.createElement("li");  
        item.appendChild(document.createTextNode("Option " + i);  
        list.appendChild(item);  
    }  
    list.style.display = "";  
     

        【3】一次性,修改样式属性。Demo如下:

    // 不好的做法
    // 这种做法会触发多次重排
    element.style.backgroundColor = "blue";  
    element.style.color = "red";  
    element.style.fontSize = "12em";
     
    // 更好的做法是,把样式都放在一个class下
    .newStyle {  
        background-color: blue;  
        color: red;  
        font-size: 12em;  
    }  
    element.className = "newStyle";
     

        【4】使用缓存,缓存临时节点。

    // 不好的做法
    document.getElementById("myDiv").style.left = document.getElementById("myDiv").offsetLeft +  
    document.getElementById("myDiv").offsetWidth + "px";  
    // 更好的做法
    var myDiv = document.getElementById("myDiv");  
    myDiv.style.left = myDiv.offsetLeft + myDiv.offsetWidth + "px";  
  • 相关阅读:
    MT【347】单变量求最值
    MT【346】拐点处分界
    MT【345】三个绝对值的和
    MT【344】构造函数
    MT【343】三数平方法
    MT【342】条件为非负实数
    MT【341】换元变形
    MT【340】彭塞列闭合定理
    MT【339】待定系数
    MT【338】分式变形
  • 原文地址:https://www.cnblogs.com/hgonlywj/p/4852973.html
Copyright © 2011-2022 走看看