js优化针对IE6.0起作用(详细整理)


Posted in Javascript onDecember 25, 2012

js优化针对IE6.0起作用,总结一下几点:

一,字符串拼接:用数组拼接

function func2(){ 
var start = new Date().getTime(); 
var array = []; 
for(var i = 0; i < 10000; i++){ 
array[i] = "<input type='button' value='a'>"; 
}

二,for 循环:先把长度算出来直接调用
function func2(){ 
var divs = document.getElementsByTagName("div"); 
var start = new Date().getTime(); 
for(var i = 0, len = divs.length; i < len; i++){ 
//"效率高" 
}

三,减少页面的重绘:可以用一中把页面拼接起来然后再赋值给页面
function func2(){ 
var obj = document.getElementById("demo"); 
var start = new Date().getTime(); 
var arr = []; 
for(var i = 0; i < 100; i++){ 
arr[i] = str + i; 
} 
obj.innerHTML = arr.join("");

四,减少作用域链上的查找次数:如果取多个页面值则定义一个document对象,再调用这个对象
var doc = document; 
for(var i = 0; i < 10000; i++){ 
var but1 = doc.getElementById("but1"); 
var but2 = doc.getElementById("but2"); 
var inputs = doc.getElementsByTagName("input"); 
} 
}

五,避免双重解释:不要重复调用函数或者方法

1、字符串的拼接

字符串的拼接在我们开发中会经常遇到,所以我把其放在首位,我们往往习惯的直接用+=的方式来拼接字符串,其实这种拼接的方式效率非常的低,我们可以用一种巧妙的方法来实现字符串的拼接,那就是利用数组的join方法。

<div class="one" id="one"></div> 
<input type="button" value="效率低" onclick="func1()" /> 
<input type="button" value="效率高" onclick="func2()" /> //效率低的 
function func1(){ 
var start = new Date().getTime(); 
var template = ""; 
for(var i = 0; i < 10000; i++){ 
template += "<input type='button' value='a'>"; 
} 
var end = new Date().getTime(); 
document.getElementById("one").innerHTML = template; 
alert("用时:" + (end - start) + "毫秒"); 
} 
//效率高的 
function func2(){ 
var start = new Date().getTime(); 
var array = []; 
for(var i = 0; i < 10000; i++){ 
array[i] = "<input type='button' value='a'>"; 
} 
var end = new Date().getTime(); 
document.getElementById("one").innerHTML = array.join(""); 
alert("用时:" + (end - start) + "毫秒"); 
}
 
我们看看其在不同浏览器下执行的情况
js优化针对IE6.0起作用(详细整理)

我们会发现,在IE6下其差别是相当明显的,其实这种情况在IE的高版本中体现的也非常明显,但是在Firefox下却没有多大的区别,相反第二种的相对效率还要低点,不过只是差别2ms左右,而Chrome也和Firefox类似。另外在这里顺便说明一下,在我们给数组添加元素的时候,很多人喜欢用数组的原生的方法push,其实直接用arr[i]或者arr[arr.length]的方式要快一点,大概在10000次循环的情况IE浏览器下会有十几毫秒的差别。

2、for循环

for循环是我们经常会遇到的情况,我们先看看下面例子:

<input type="button" value="效率低" onclick="func1()" /> 
<input type="button" value="效率高" onclick="func2()" /> 
var arr = []; 
for(var i = 0; i < 10000; i++){ 
arr[i] = "<div>" + i + "</div>"; 
} 
document.body.innerHTML += arr.join(""); //效率低的 
function func1(){ 
var divs = document.getElementsByTagName("div"); 
var start = new Date().getTime(); 
for(var i = 0; i < divs.length; i++){ 
//"效率低" 
} 
var end = new Date().getTime(); 
alert("用时:" + (end - start) + "毫秒"); 
} 
//效率高的 
function func2(){ 
var divs = document.getElementsByTagName("div"); 
var start = new Date().getTime(); 
for(var i = 0, len = divs.length; i < len; i++){ 
//"效率高" 
} 
var end = new Date().getTime(); 
alert("用时:" + (end - start) + "毫秒"); 
}
js优化针对IE6.0起作用(详细整理)

由上表可以看出,在IE6.0下,其差别是非常明显,而在Firefox和Chrome下几乎没有差别,之所以在IE6.0下会有这种情况,主要是因为for循环在执行中,第一种情况会每次都计算一下长度,而第二种情况却是在开始的时候计算长度,并把其保存到一个变量中,所以其执行效率要高点,所以在我们使用for循环的时候,特别是需要计算长度的情况,我们应该开始将其保存到一个变量中。但是并不是只要是取长度都会出现如此明显的差别,如果我们仅仅是操作一个数组,取得的是一个数组的长度,那么其实两种方式的写法都差不多,我们看下面的例子:

<input type="button" value="效率低" onclick="func1()" /> 
<input type="button" value="效率高" onclick="func2()" /> 
var arr2 = []; 
for(var i = 0; i < 10000; i++){ 
arr2[i] = "<div>" + i + "</div>"; 
} 
//效率低的 
function func1(){ 
var start = new Date().getTime(); 
for(var i = 0; i < arr2.length; i++){ 
//"效率低" 
} 
var end = new Date().getTime(); 
alert("用时:" + (end - start) + "毫秒"); 
} 
//效率高的 
function func2(){ 
var start = new Date().getTime(); 
for(var i = 0, len = arr2.length; i < len; i++){ 
//"效率高" 
} 
var end = new Date().getTime(); 
alert("用时:" + (end - start) + "毫秒"); 
}
js优化针对IE6.0起作用(详细整理)

从上表可以看出,如果仅仅是一个数组的话,我们看到其实两种写法都是差不多的,其实如果我们把循环再上调到100000次的话,也仅仅是差别几毫秒而已,所以在数组的情况下,我认为都是一样的。对于for循环的优化,也有人提出很多点,有人认为用-=1,或者从大到小的方式循环等等,我认为是完全没有必要的,这些优化往往实际情况下根本没有表现出来,换句话说只是计算机级别的微小的变化,但是给我们带来的却是代码的可读性大大的降低,所以实在是得不偿失。

3、减少页面的重绘

减少页面重绘虽然本质不是JS本身的优化,但是其往往是由JS引起的,而重绘的情况往往是严重影响页面性能的,所以完全有必要拿出来,我们看下面例子:

<div id="demo"></div> 
<input type="button" value="效率低" onclick="func1()" /> 
<input type="button" value="效率高" onclick="func2()" /> 
var str = "<div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div><div>这是一个测试字符串</div>"; 
//效率低的 
function func1(){ 
var obj = document.getElementById("demo"); 
var start = new Date().getTime(); 
for(var i = 0; i < 100; i++){ 
obj.innerHTML += str + i; 
} 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
} 
//效率高的 
function func2(){ 
var obj = document.getElementById("demo"); 
var start = new Date().getTime(); 
var arr = []; 
for(var i = 0; i < 100; i++){ 
arr[i] = str + i; 
} 
obj.innerHTML = arr.join(""); 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
}

在例子中,我只是用了100次的循环,因为如果用10000次循环的话,浏览器基本上就卡住不动了,但是即使是100次的循环,我们看看下面的执行结果。
js优化针对IE6.0起作用(详细整理)

可以看到的是,这简直是一个惊人的结果,仅仅100次的循环,不管是在什么浏览器下,都出现了如此之大的差别,另外我们还发现,在这里,IE6的执行效率居然比起Firefox还要好很多,可见Firefox在页面重绘这方面并没有做一些的优化。这里还要注意的是,一般影响页面重绘的不仅仅是innerHTML,如果改变元素的样式,位置等情况都会触发页面重绘,所以在平时一定要注意这点。

4、减少作用域链上的查找次数

我们知道,js代码在执行的时候,如果需要访问一个变量或者一个函数的时候,它需要遍历当前执行环境的作用域链,而遍历是从这个作用域链的前端一级一级的向后遍历,直到全局执行环境,所以这里往往会出现一个情况,那就是如果我们需要经常访问全局环境的变量对象的时候,我们每次都必须在当前作用域链上一级一级的遍历,这显然是比较耗时的,我们看下面的例子:

<div id="demo"></div> 
<input id="but1" type="button" onclick="func1()" value="效率低"/> 
<input id="but2" type="button" onclick="func2()" value="效率高"/> function func1(){ 
var start = new Date().getTime(); 
for(var i = 0; i < 10000; i++){ 
var but1 = document.getElementById("but1"); 
var but2 = document.getElementById("but2"); 
var inputs = document.getElementsByTagName("input"); 
var divs = document.getElementsByTagName("div"); 
var but1 = document.getElementById("but1"); 
var but2 = document.getElementById("but2"); 
var inputs = document.getElementsByTagName("input"); 
var divs = document.getElementsByTagName("div"); 
var but1 = document.getElementById("but1"); 
var but2 = document.getElementById("but2"); 
var inputs = document.getElementsByTagName("input"); 
var divs = document.getElementsByTagName("div"); 
var but1 = document.getElementById("but1"); 
var but2 = document.getElementById("but2"); 
var inputs = document.getElementsByTagName("input"); 
var divs = document.getElementsByTagName("div"); 
var but1 = document.getElementById("but1"); 
var but2 = document.getElementById("but2"); 
var inputs = document.getElementsByTagName("input"); 
var divs = document.getElementsByTagName("div"); 
var but1 = document.getElementById("but1"); 
var but2 = document.getElementById("but2"); 
var inputs = document.getElementsByTagName("input"); 
var divs = document.getElementsByTagName("div"); 
} 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
} 
function func2(){ 
var start = new Date().getTime(); 
var doc = document; 
for(var i = 0; i < 10000; i++){ 
var but1 = doc.getElementById("but1"); 
var but2 = doc.getElementById("but2"); 
var inputs = doc.getElementsByTagName("input"); 
var divs = doc.getElementsByTagName("div"); 
var but1 = doc.getElementById("but1"); 
var but2 = doc.getElementById("but2"); 
var inputs = doc.getElementsByTagName("input"); 
var divs = doc.getElementsByTagName("div"); 
var but1 = doc.getElementById("but1"); 
var but2 = doc.getElementById("but2"); 
var inputs = doc.getElementsByTagName("input"); 
var divs = doc.getElementsByTagName("div"); 
var but1 = doc.getElementById("but1"); 
var but2 = doc.getElementById("but2"); 
var inputs = doc.getElementsByTagName("input"); 
var divs = doc.getElementsByTagName("div"); 
var but1 = doc.getElementById("but1"); 
var but2 = doc.getElementById("but2"); 
var inputs = doc.getElementsByTagName("input"); 
var divs = doc.getElementsByTagName("div"); 
var but1 = doc.getElementById("but1"); 
var but2 = doc.getElementById("but2"); 
var inputs = doc.getElementsByTagName("input"); 
var divs = doc.getElementsByTagName("div"); 
} 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
}

上面代码中,第二种情况是先把全局对象的变量放到函数里面先保存下来,然后直接访问这个变量,而第一种情况是每次都遍历作用域链,直到全局环境,我们看到第二种情况实际上只遍历了一次,而第一种情况却是每次都遍历了,所以我们看看其执行结果:
js优化针对IE6.0起作用(详细整理)

从上表中可以看出,其在IE6下差别还是非常明显的,而且这种差别在多级作用域链和多个全局变量的情况下还会表现的非常明显。

5、避免双重解释

双重解释的情况也是我们经常会碰到的,有的时候我们没怎么考虑到这种情况会影响到效率,双重解释一般在我们使用eval、new Function和setTimeout等情况下会遇到,我们看看下面的例子:

<div id="demo"></div> 
<input id="but1" type="button" onclick="func1()" value="效率低"/> 
<input id="but2" type="button" onclick="func2()" value="效率高"/> 
var sum, num1 = 1, num2 = 2; 
function func1(){ 
var start = new Date().getTime(); 
for(var i = 0; i < 10000; i++){ 
var func = new Function("sum+=num1;num1+=num2;num2++;"); 
func(); 
} 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
} function func2(){ 
var start = new Date().getTime(); 
for(var i = 0; i < 10000; i++){ 
sum+=num1; 
num1+=num2; 
num2++; 
} 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
}

第一种情况我们是使用了new Function来进行双重解释,而第二种是避免了双重解释,我们看看在不同浏览器下的表现:
js优化针对IE6.0起作用(详细整理)

可以看到,在所有的浏览器中,双重解释都是有很大开销的,所以在实际当中要尽量避免双重解释。

感谢"SeaSunK"对第四点测试报告错误的指正,现在已经修改过来了。至于最后一点提出的func1每次都初始化,没有可比性,所以我给换了eval,结果发现,在IE6.0下还是有影响,而且在Firefox下,使用eval对效率的影响程度更加厉害,在Firefox下,如果10000次循环,需要十多秒的时间,所以我把循环都变成了1000次。看代码和报告。

var sum, num1 = 1, num2 = 2; 
function func1(){ 
var start = new Date().getTime(); 
for(var i = 0; i < 1000; i++){ 
eval("sum+=num1;num1+=num2;num2++;"); 
} 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
} 
function func2(){ 
var start = new Date().getTime(); 
for(var i = 0; i < 1000; i++){ 
sum+=num1; 
num1+=num2; 
num2++; 
} 
var end = new Date().getTime(); 
alert("用时 " + (end - start) + " 毫秒"); 
}
Javascript 相关文章推荐
不能再简单的无闪刷新验证码原理很简单
Nov 05 Javascript
javascript下arguments,caller,callee,call,apply示例及理解
Dec 24 Javascript
Jquery实战_读书笔记2 选择器
Jan 22 Javascript
javascript小数四舍五入多种方法实现
Dec 23 Javascript
Node.js中使用事件发射器模式实现事件绑定详解
Aug 15 Javascript
JavaScript仿微博输入框效果(案例分析)
Dec 06 Javascript
Angular2入门--架构总览
Mar 29 Javascript
AngularJS入门教程二:在路由中传递参数的方法分析
May 27 Javascript
Angular实现类似博客评论的递归显示及获取回复评论的数据
Nov 06 Javascript
JavaScript生成一个不重复的ID的方法示例
Sep 16 Javascript
React中Ref 的使用方法详解
Apr 28 Javascript
如何在现代JavaScript中编写异步任务
Jan 31 Javascript
js 取时间差去掉周六周日实现代码
Dec 25 #Javascript
纯js网页画板(Graphics)类简介及实现代码
Dec 24 #Javascript
Knockoutjs快速入门(经典)
Dec 24 #Javascript
Underscore.js 的模板功能介绍与应用
Dec 24 #Javascript
javascript 星级评分效果(手写)
Dec 24 #Javascript
Jquery写一个鼠标拖动效果实现原理与代码
Dec 24 #Javascript
jquery限制输入字数,并提示剩余字数实现代码
Dec 24 #Javascript
You might like
收音机史话 - 1960年代前后的DIY
2021/03/02 无线电
thinkphp项目部署到Linux服务器上报错“模板不存在”如何解决
2016/04/27 PHP
php5.2的curl-bug 服务器被php进程卡死问题排查
2016/09/19 PHP
php版微信发红包接口用法示例
2016/09/23 PHP
Smarty日期时间操作方法示例
2016/11/15 PHP
PHP三种方式实现链式操作详解
2017/01/21 PHP
Laravel框架中Blade模板的用法示例
2017/08/30 PHP
phpstudy的php版本自由修改的方法
2017/10/18 PHP
类似php的js数组的in_array函数自定义方法
2013/12/27 Javascript
扩展IE中一些不兼容的方法如contains、startWith等等
2014/01/09 Javascript
EasyUI实现第二层弹出框的方法
2015/03/01 Javascript
js+jquery常用知识点汇总
2015/03/03 Javascript
jQuery  ready方法实现原理详解
2016/10/19 Javascript
JavaScript类型相关的常用操作总结
2019/02/14 Javascript
vue实现修改图片后实时更新
2019/11/14 Javascript
Python开发WebService系列教程之REST,web.py,eurasia,Django
2014/06/30 Python
总结Python中逻辑运算符的使用
2015/05/13 Python
Python实现的RSS阅读器实例
2015/07/25 Python
django模板加载静态文件的方法步骤
2019/03/01 Python
Django如何开发简单的查询接口详解
2019/05/17 Python
python for循环remove同一个list过程解析
2019/08/14 Python
详解在Python中以绝对路径或者相对路径导入文件的方法
2019/08/30 Python
基于Python实现人脸自动戴口罩系统
2020/02/06 Python
keras中模型训练class_weight,sample_weight区别说明
2020/05/23 Python
Matlab中plot基本用法的具体使用
2020/07/17 Python
Python实现网络聊天室的示例代码(支持多人聊天与私聊)
2021/01/27 Python
使用CSS3编写类似iOS中的复选框及带开关的按钮
2016/04/11 HTML / CSS
Wallis官网:英国女装零售商
2020/01/21 全球购物
c++工程师面试问题
2013/08/04 面试题
杭州时比特电子有限公司SQL
2013/08/22 面试题
幼儿园元旦亲子活动方案
2014/02/17 职场文书
工会2014法制宣传日活动总结
2014/11/01 职场文书
师德承诺书
2015/01/20 职场文书
档案管理员岗位职责
2015/02/12 职场文书
详解SQL的窗口函数
2022/04/21 Oracle
Java实现添加条码或二维码到Word文档
2022/06/01 Java/Android