对Dom的访问代价是昂贵,在富网页应用中通常是性能的瓶颈,所以对Dom的优化十分重要。

一.访问和修改Dom元素

浏览器通常要求JavaScript和Dom实现保持独立的。例如,在Internet Explorer 中,被称为JScript的JavaScript 实现位于库文件jscript.dll 中,而DOM 实现位于另一个库mshtml.dll(内部代号Trident)。所以用JavaScript访问Dom元素是需要一定的代价。人们通常这样形容Dom访问,javaScript和Dom相当于两个独立的小岛,这两个小岛中间有座桥连起来,JavaScript每次访问Dom都要过一次桥,过桥就要收取一定的过桥费。所以要尽量减少访问Dom的次数,那么如何做到呢?请看下面的例子。

  function innerHTMLLoop() {
    for (var count = 0; count < 15000; count++) {
      document.getElementById('here').innerHTML += 'a';
    }
  }

上面这个函数的作用是对id是here这个Dom节点,进行了15000次修改,在这个函数中,对Dom节点进行了15000次访问。再看下面的函数。

  function innerHTMLLoop() {

var text="";
    for (var count = 0; count < 15000; count++) {
       text+= 'a';
    }

document.getElementById('here').innerHTML=text;
  }

上面这个函数和第一个函数实现的是完全一样的功能,但是它对Dom的访问只有一次,性能是大大提高。

现在来比较一下document.createElement和innerHTML。

document.createElement是每次创建一个节点,然后加入到父节点中。innerHTML是用字符串拼接成一系列Dom节点然后在插入到目标节点中。如果在一个性能苛刻的操作中更新一大块HTML 页面,innerHTML 在大多数浏览器中执行更快。但对于大多数日常操作而言,其差异并不大,所以你应当根据代码可读性,可维护性,团队习惯,代码风格来综合决定采用哪种方法。

另外HTML集合用于存放Dom节点类数组对象,例如document.getElementsByTagName,getElementsByClass等函数得到的就是集合。它的使用也是非常昂贵的。考虑下面的例子

  var alldivs = document.getElementsByTagName_r('div');
  for (var i = 0; i < alldivs.length; i++) {
    document.body.appendChild(document.createElement('div'))
  }

上面的例子表面上是想倍增div的数量,但实际上是个死循环,因为div的长度是动态变化的,也就是说alldivs.length不是一个常量,它会每次根据div的数量变化,这样也就是每次要访问Dom节点,我们之前说过访问Dom的代价是昂贵的。所以最好的方法是将length用一个变量缓存起来,例如len=alldivs.length。

 二.重绘和重排

我们都知道在页面刚加载的时候,浏览器会进行页面渲染,其实这就是一次绘制和排列的过程。绘制的是页面,排列的是Dom节点,一般来说绘制是根据color,background,opacity等css属性,而排列则根据width,height,margin,padding等能影响页面Dom元素位置变换的css属性。那么既然知道了什么是重绘和重排,就应该知道什么时候会重会和重排。也就是当你改变某些css属性时。

一般来说重排的影响是远大于重绘的,重排时浏览器会重新计算所有元素的位置和尺寸,重新排列。这个过程可以说是相当耗时间的,所以要尽量避免重排。考虑下面例子

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

上面例子三次改变了el的css属性,而这三次都产生的重排。再看下面代码

var el = document.getElementById('mydiv');

el.style.cssText="border-left:1px;border-right:2px;padding:5px";

上面代码实现的是和第一例子一样的功能,但是它只产生的一次重排,大大提高了性能。

 三.事件托管

  当页面中存在大量元素,而且每个元素有一个或多个事件句柄与之挂接(例如onclick)时,可能会影响性能。连接每个句柄都是有代价的,无论其形式是加重了页面负担(更多的页面标记和JavaScript 代码)还是表现在运行期的运行时间上。所以事件托管是很有用的。

每个事件发生时会经历捕获,到达目标,冒泡,三个阶段。例如:

 

  当用户点击了“menu #1”链接,点击事件首先被<a>元素收到。然后它沿着DOM 树冒泡,被<li>元素收到,然后是<ul>,接着是<div>,等等,一直到达文档的顶层,甚至window。如果需要在每个li上加监听事件,那么可以加到ul上,因为每次点击li都会冒泡到ul上。

  总结:

DOM 访问和操作是现代网页应用中很重要的一部分。但每次你通过桥梁从ECMAScript 岛到达DOM 岛时,都会被收取“过桥费”。为减少DOM 编程中的性能损失,请牢记以下几点:

最小化DOM 访问,在JavaScript 端做尽可能多的事情

在反复访问的地方使用局部变量存放DOM 引用.  

  小心地处理HTML 集合,因为他们表现出“存在性”,总是对底层文档重新查询。将集合的length 属性缓存到一个变量中,在迭代中使用这个变量。如果经常操作这个集合,可以将集合拷贝到数组中。 
    注意重绘和重排版;批量修改风格,离线操作DOM 树,缓存并减少对布局信息的访问。

使用事件托管技术最小化事件句柄数量。

Dom编程优化的更多相关文章

  1. 性能优化之数据存储&DOM编程

    多读书多看报 数据存储 ·在javascript中,数据存储的位置会对代码整体性能产生重大的影响. ·数据存储共有4种方式:字面量.变量.数组.对象成员.   ·要理解变量的访问速度,就要理解作用域. ...

  2. JavaScript性能优化 DOM编程

    最近在研读<高性能JavaScript>,在此做些简单记录.示例代码可在此处查看到. 一.DOM 1)DOM和JavaScript 文档对象模型(DOM)是一个独立于语言的,用于操作XML ...

  3. 《高性能javascript》 领悟随笔之-------DOM编程篇

    <高性能javascript> 领悟随笔之-------DOM编程篇一 序:在javaSctipt中,ECMASCRIPT规定了它的语法,BOM实现了页面与浏览器的交互,而DOM则承载着整 ...

  4. 高性能JavaScript笔记一(加载和执行、数据访问、DOM编程)

    写在前面 好的书,可能你第一遍并不能领会里面的精魂,当再次细细品评的时候,发现领悟的又是一层新的含义 (这段时间,工作上也不会像从前一样做起来毫不费力,开始有了新的挑战,现在的老大让我既佩服又嫉妒,但 ...

  5. 160826、浏览器渲染页面过程描述,DOM编程技巧以及重排和重绘

    一.浏览器渲染页过程描述   1.浏览器解析html源码,然后创建一个DOM树. 在DOM树中,每一个HTML标签都有一个对应的节点(元素节点),并且每一个文本也都有一个对应的节点(文本节点). DO ...

  6. 浏览器渲染页面过程描述,DOM编程技巧以及重排和重绘。

    一.浏览器渲染页过程描述 1.浏览器解析html源码,然后创建一个DOM树. 在DOM树中,每一个HTML标签都有一个对应的节点(元素节点),并且每一个文本也都有一个对应的节点(文本节点). DOM树 ...

  7. 《javascript dom编程艺术》笔记(一)——优雅降级、向后兼容、多个函数绑定onload函数

    刚刚开始自学前端,如果不对请指正:欢迎各位技术大牛指点. 开始学习<javascript dom编程艺术>,整理一下学习到的知识.今天刚刚看到第六章,记下get到的几个知识点. 优雅降级 ...

  8. 高性能JavaScript之DOM编程

    我们知道.DOM是用于操作XML和HTML文档的应用程序接口,用脚本进行DOM操作的代价非常昂贵. 有个贴切的比喻.把DOM和JavaScript(这里指ECMScript)各自想象为一个岛屿,它们之 ...

  9. 高性能Javascript(2) DOM编程

    第三部分 DOM编程 文档对象模型(DOM)是一个独立于语言的,使用XML和HTML文档操作的应用程序接口(API).在浏览器中,主要与HTML文档打交道,在网页应用中检索XML文档也很常见.DOM ...

随机推荐

  1. shell整数运算

  2. KiCAD原理图导出PDF方法

    KiCAD原理图导出为PDF 1.文件->绘制 2.按照下图选择保存目录和输出格式后,选择绘制当前页或者所有页

  3. 每天一个Linux常用命令 cp命令

    Linux cp命令主要用于复制文件或目录 -a:此选项通常在复制目录时使用,它保留链接.文件属性,并复制目录下的所有内容.其作用等于dpR参数组合. -d:复制时保留链接.这里所说的链接相当于Win ...

  4. MySQL源码编译与初始化

    MySQL源码编译与初始化 链接:https://pan.baidu.com/s/1ANGg3Kd_28BzQrA5ya17fQ 提取码:ekpy 复制这段内容后打开百度网盘手机App,操作更方便哦 ...

  5. Codeforces 454E. Little Pony and Summer Sun Celebration

    题意:给n个点m条边的无向图,并给出每个点的访问次数奇偶,求构造一条满足条件的路径(点和边都可以走). 解法:这道题还蛮有意思的.首先我们可以发现在一棵树上每个儿子的访问次数的奇偶是可以被它的父亲控制 ...

  6. tomcat 安装时出现 Failed to install Tomcat7 service

    今天在安装tomcat时提示 Failed to install Tomcat7 service了,花了大半天的时间找到了原因,下面分享给大家,希望对各位有所帮助. 应该是你卸载时直接删除目录导致的. ...

  7. Windows系统文件夹共享与隐藏共享

    Windows用户之间建立共享文件夹提供共享服务 建立共享文件夹 Window共享:本地网络-->属性-->共享-->高级共享   设置共享名-->另一台主机输入共享主机的IP ...

  8. kubernetes(k8s)Pod污点与容忍

    污点(taints)与容忍(tolerations) 对于nodeAffinity无论是硬策略还是软策略方式,都是调度 pod 到预期节点上,而Taints恰好与之相反,如果一个节点标记为 Taint ...

  9. AcWing 160. 匹配统计 (哈希+二分) 打卡

    阿轩在纸上写了两个字符串,分别记为A和B. 利用在数据结构与算法课上学到的知识,他很容易地求出了“字符串A从任意位置开始的后缀子串”与“字符串B”匹配的长度. 不过阿轩是一个勤学好问的同学,他向你提出 ...

  10. NX二次开发-NXOpenC++ Example

    NxOpenC++ Example NXOpen::WCS wcs坐标系 https://www.cnblogs.com/nxopen2018/p/11368763.html NXOpen::Draw ...