做了这么久的前端工程师,总被朋友问到怎么写出高性能的javascript,那么我今天就来简单总结下,其实js本身是没有什么性能问题的,所谓的内存泄露,也主要针对于IE6,IE7,而IE7的内存泄露问题也并不严重,这里不讨论浏览器造成的内存泄露问题,我们只讨论,养成什么样的书写习惯能够写出,高效率高性能的js。
在这里我总结了三个书写js的习惯,然后分别针对执行效率,内存问题,安全等各个方面综合分析要养成这三个习惯的原因。
1.尽量使用局部变量:
局部变量的创建和访问都是特别廉价快捷的,而使用全局变量的话,js对全局变量的调用实际上是对GLOBAL对象的查找引用,性能低下。 而局部变量则是直接创建于当前作用域,不需要有查找引用的过程。不单单是当前作用域,包括闭包里的局部变量访问,都是很快的。不过,这个过程带来的优化是需要庞大的js程序长时间的运行才能体现出的。
下面通过细化js解析过程来理解下:
一个变量具体是局部变量还是全局变量,局部变量的话是在哪层闭包里,是第几个变量,在编译的时候就已经确定了,重复的var声明并不会影响js的执行效率,js执行时,如果是局部变量则会直接在向内存地址里取用,而全局变量则是对象访问,显然局部变量在效率上要优化的多。
但是,不是任何时候局部变量都是最优化的。
例如,当一个变量是一个表达式或者一个查找dom节点的过程等时,变量的取用中就带有了查找和计算过程,这样如果计算次数比较多,用全局变量则可以一次性的固化结果,而使用局部变量反而需要多次计算查找,而影响了效率了。
总结:
当然,以上的数据仅仅只对js代码的执行效率而言,在具体应用中,我仍然推荐大家尽量使用局部变量。因为,js的一个重要的编程理念就是不要污染全局对象,全局对象可能被不同的模块(甚至页面上的广告)访问,在上面存放数据、函数会有不可预期的后果。所以,我们要灵活使用全局对象和局部变量。
2.及时释放回调函数,解除事件绑定。
上面一个习惯,主要针对于执行效率问题,而这个习惯则是针对内存泄露的问题了。除了古老浏览器本身的bug造成的内存泄露外,错误的书写js代码也会引起内存泄露的问题。
首先,有如下代码解释下这句话的意思:
1 img.onload = function () { 2 img.onload = img.onerror = null; 3 } 4 $(...).on('keyup', function(e){ 5 $().off(e); 6 })
那这么做有什么好处呢,那么我们看下下面的情况:
3.及时清除引用。
这个习惯也是针对于内存泄露问题,我们先看看以下代码:
1 <!doctype html> 2 <html> 3 <head> 4 <title>Memory leak demo</title> 5 </head> 6 <body> 7 <a href="javascript:;">Click me</a> 8 </body> 9 <script src="jquery-1.8.0.js"></script><script> 10 $('a').click(function callback(e) { 11 $(this).remove(); 12 $('<a href="javascript:;">Click me</a>').appendTo(document.body).click(function(e2){ 13 e2.leak = e; 14 callback.call(this, e2); 15 }); 16 17 var current = e, n = 0; 18 while(current.leak){ 19 current = current.leak; 20 n++; 21 } 22 console.log('leaked: ' + n); 23 //e = null; 24 }); 25 </script> 26 </html>
这是段代码在不停的替换页面中的a标签,并且为新的a标签绑定clcik事件,在事件的回调函数中,引用了上一个事件的e,导致了原本的a标签即使已经被remove也无法被释放,仍然保存在内存里。随着用户click次数的增多,内存泄露问题也会越来越严重。
所以这个时候,我们就要记得及时清除引用,这段代码的内存泄露问题是由于事件对象e引起的,在23行,去掉注释后,e = null 释放了内层函数对外层函数的变量e指向的对象的引用,使得e以及它引用的dom节点能被释放,内存泄露问题也就得以解决了。
当然,想写出高性能的js,还有很多可以仔细研究的地方,而上面的三个习惯是可以在日常的编码中养成的,也没有什么替换成本。还有很多良好的习惯,比如减少函数引用,尽量少使用闭包,等,想做到这些方式可能需要较高的替换成本,如果有机会以后再与大家分享吧~。