dom操作中的js优化

频繁地对于DOM进行操作的很是损耗性能,但在富网页应用中我们编写脚本无可避免地要跟DOM打交道,到底怎么才能优化这个性能瓶颈呢,大致从以下三种情况去考虑:

访问和修改DOM元素

修改DOM样式,会造成页面的重绘和重新排版

通过DOM事件处理程序来响应用户

访问和修改DOM元素

  在浏览器中,DOM的实现和Javascript的实现通常是保持相互独立的。下面了解一下主流浏览器的渲染引擎和JS引擎:

  浏览器     渲染引擎(内核)     JS引擎  
IE   mshtml.dll(Trident)   JScript
Chrome   WebCore(WebKit)   V8
FireFox   Gecko    Spider-Monkey/TraceMonkey 
Safari   WebCore(WebKit)    JavaScriptCore/SquirrelFish 

访问DOM元素的代价就是交一次“桥费”,修改DOM元素则会导致浏览器重新计算页面的几何变化。如果是循环修改DOM元素,其代价可想而知。如下代码:



function innerHTMLLoop1(){
    for(var count=0;count<10000;count++){
      document.getElementById("test").innerHTML +="增加内容";
    }
}

  在这段代码中,每循环一次都要对DOM元素访问两次:一次是读取innerHTML属性的内容,另一次是把新的内容写入它。所以一个优化的办法就是使用一个局部变量存储更新后的内容,在循环结束时再一次性写入:

function innerHTMLLoop2(){  var content ="";
   for(var count=0;count<10000;count++){
      content +="增加内容";
    }
   document.getElementById("test").innerHTML += content ;
}

很明显,innerHTMLLoop2交的“桥费”明显要少,因为它只访问了两次DOM元素,一次读入,一次写入。

更新页面的两种方法性能比较:innerHTML和DOM方法(如document.creatElment)。性能差别不大,innerHTML好一些,使用简单嘛。另外还有一个更新页面的方法是节点克隆--element.cloneNode()。

HTML集合的操作

HTML集合是用于存放DOM节点引用的类数组对象。可通过下列的方法或属性得到这样的集合:

document.getElementsByName()
document.getElementsByTagName()
document.getElementsByClassName()
document.images 返回对文档中所有 Image 对象引用
document.links  返回对文档中所有 Area 和 Link 对象引用
document.forms 返回对文档中所有 Form 对象引用
document.forms[0].elements 返回对文档中第一个表单的所有元素

HTML集合会实时查询文档信息,也就是说当你要用到这个集合时,它会自动查询文档的最新信息。请看如下代码:

var allDivs = document.getElementsByTagName("div");   
for(var i=0;i<allDivs.length;i++){
       document.body.appendChild(document.createElement("div"));   
}

例如像上面的那段代码其实是个死循环,因为每一次访问div集合的length属性,它都会重新计算文档中的div元素数目。这就是html集合低效率的来源。要改进代码就用一个局部变量保存div集合的length属性:

for(var i=0, len=allDivs.length; i<len; i++){...}

访问HTML集合的length比数组的length要慢,所以要访问这种集合类的数目length,我们都应该先用一个局部变量去保存它:

var len = 集合.length;

另外,访问数组的元素要比访问HTML集合的元素要快。所以我们可以先把HTML集合转换成数组才去进行相应的操作:

//HTML集合转换成数组  
function toArray(coll){
    for(var a=[], i=0, len=coll.legnth; i<len; i++){
      a[i] = coll[i];
    }
    return a;
  }  
//使用  
var coll = document.getElementsByTagName("div");  
var divs = toArray(coll);

有人可能会问,这样多用了一个数组副本到底值不值得?这个倒是要看情况吧。不过另外一种选择,使用局部变量:
function loopColletion(){
    var coll = document.getElmentsByTagName("div");
    var len = coll.length;
    var el = null;
    for(var i = 0; i<len; i++){
      el = coll[i];
      //然后访问局部变量el
    }
  }

许多浏览器提供了API函数返回元素节点,这些API都是原生的,所以可用的话就尽量用。下图列举了一些DOM的属性:


上图列举的所有属性能被FF,safari,chrome,opera所支持,ie6-8只支持children。

  遍历children比childNodes更快,因为集合项少了。HTML源码中的空格实际上是文本节点,但他们不包含在children中。

另外还有两个比较好的选择器API:document.querySelectorAll()和document.querySelector()。前者接收一个CSS选择器字符串参数并返回一个NodeList类数组对象而不是返回HTML集合,后者只返回符合查询条件的第一个节点。很遗憾IE6、7不支持这两个API。

时间: 2024-08-05 21:08:07

dom操作中的js优化的相关文章

js优化原则

首先,与其他语言不同,JS的效率很大程度是取决于JS engine的效率.除了引擎实现的优劣外,引擎自己也会为一些特殊的代码模式采取一些优化的策略.例如FF.Opera和Safari的JS引擎,都对字符串的拼接运算(+)做了特别优化.显然,要获得最大效率,就必须要了解引擎的脾气,尽量迎合引擎的口味.所以对于不同的引擎,所作的优化极有可能是背道而驰的. 而如果做跨浏览器的web编程,则最大的问题是在于IE6(JScript 5.6)!因为在不打hotfix的情况下,JScript引擎的垃圾回收的b

js 优化方面

根据网上查阅关于优化方面的问题,根据自己理解,进行简单整理.仅供大家在优化方面参考 一.在web开发中,通过dom 对浏览器进行操作,来实现js 与html 的交互,而DOM的操作是非常昂贵的,因为这会导致浏览器执行回流 (reflow)操作.但我们不可避免进行dom操作,所以选择尽可能的减少DOM操作来进行优化. 回流(是指浏览器为了重新渲染部分或者全部的文档而重新计算文档中元素的位置和几何构造的过程.)所以通过操作引起元素位置的改变就会引起回流. 导致回流产生的情况:  用户的操作,比如改变

前端js优化方案(连续更新)

最近在读<高性能javascript>,在这里记录一下读后的一些感受,顺便加上自己的一些理解,如果有兴趣的话可以关注的我的博客http://www.bloggeng.com/,我会不定期发布一些前端教程还有前端干货,话不多说,进入我们的正题 一.js优化之加载 1.script标签放在底部,不要放在head标签: 2.尽量减少js文件数量 3.给script标签添加一个defer属性,defer属性指明所含的脚本不会修改DOM,因此代码能安全地延迟执行. <script type=&qu

DOM操作的性能优化

p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 12.0px "PingFang SC"; color: #454545 } span.s1 { font: 12.0px Helvetica } DOM操作的真正问题在于 每次操作都会出发布局的改变.DOM树的修改和渲染. React解决了大面积的DOM操作的性能问题,实现了一个虚拟DOM,即virtual DOM,这个我们一条条讲. 所以关于DOM操作的性能优化主要包括: 1.查找元素的优化

DOM对象和js对象以及jQuery对象的区别

一.DOM对象 文档对象模型简称DOM,是W3C组织推荐的处理可扩展置标语言的标准编程接口. DOM实际上是以面向对象方式描述的文档模型.DOM定义了表示和修改文档所需的对象.这些对象的行为和属性以及这些对象之间的关系. 通过DOM,可以访问所有的 HTML 元素,连同它们所包含的文本和属性.可以对其中的内容进行修改和删除,同时也可以创建新的元素. HTML DOM 独立于平台和编程语言.它可被任何编程语言诸如 Java.JavaScript 和 VBScript 使用. DOM对象,即是我们用

svg DOM的一些js操作

这是第一个实例,其中讲了如何新建svg,添加元素,保存svg document,查看svg. 下面将附上常用一些元素的添加方法:(为js的,但基本上跟java中操作一样,就是类名有点细微差别) Circle var svgns = "http://www.w3.org/2000/svg"; function makeShape(evt) { if ( window.svgDocument == null ) svgDocument = evt.target.ownerDocument;

Joomla T3模板启用CSS、JS优化

如何优化生成后的文件 当启用了优化选项,CSS和JS优化文件将在一个页面第一次被加载时自动生成.如果CSS或JS有改动,新生成的优化文件也会自动生成. 生成优化文件路径: XXX/t3-assets/css XXX/t3-assets/js XXX:项目名称

r.js优化前端(摘抄)

r.js是requireJS的优化(Optimizer)工具,可以实现前端文件的压缩与合并,在requireJS异步按需加载的基础上进一步提供前端优化,减小前端文件大小.减少对服务器的文件请求.要使用r.js需下载r.js文件(点我下载),将其放到你的项目根目录:还需要安装nodeJS(点我下载),以便通过命令行来执行r.js功能.我们将拿一个小案例来详细说明使用r.js的方法(你可以点此下载这个案例). 如下图所示的项目(见案例中的before文件夹)仅仅使用了requireJS,但还未使用过

03 DOM对象(js对象)与jQuery对象

<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title></title> </head> <body> <ul> <li id="cloth">衣服</li> <li>裤子</li> <li>袜子&