当前位置:Gxlcms > JavaScript > 详谈JavaScript函数闭包使用及注意事项代码详解

详谈JavaScript函数闭包使用及注意事项代码详解

时间:2021-07-01 10:21:17 帮助过:5人阅读

闭包的英文单词是closure,这是JavaScript中非常重要的一部分知识,因为使用闭包可以大大减少我们的代码量,使我们的代码看上去更加清晰等等,总之功能十分强大。

闭包的含义:闭包说白了就是函数的嵌套,内层的函数可以使用外层函数的所有变量,即使外层函数已经执行完毕(这点涉及JavaScript作用域链)。

示例一

  1. function checkClosure(){
  2. var str = 'rain-man';
  3. setTimeout(
  4. function(){ alert(str); } //这是一个匿名函数
  5. , 2000);
  6. }
  7. checkClosure();

这个例子看上去十分的简单,仔细分析下它的执行过程还是有许多知识点的:checkClosure函数的执行是瞬间的(也许用时只是0.00001毫秒),在checkClosure的函数体内创建了一个变量str,在checkClosure执行完毕之后str并没有被释放,这是因为setTimeout内的匿名函数存在这对str的引用。待到2秒后函数体内的匿名函数被执行完毕,str才被释放。

示例二,优化代码

  1. function forTimeout(x, y){
  2. alert(x + y);
  3. }
  4. function delay(x , y , time){
  5. setTimeout('forTimeout(' + x + ',' + y + ')' , time);
  6. }
  7. /**
  8. * 上面的delay函数十分难以阅读,也不容易编写,但如果使用闭包就可以让代码更加清晰
  9. * function delay(x , y , time){
  10. * setTimeout(
  11. * function(){
  12. * forTimeout(x , y)
  13. * }
  14. * , time);
  15. * }
  16. */

匿名函数最大的用途是创建闭包(这是JavaScript语言的特性之一),并且还可以构建命名空间,以减少全局变量的使用。

示例三:

  1. var oEvent = {};
  2. (function(){
  3. var addEvent = function(){ /*代码的实现省略了*/ };
  4. function removeEvent(){}
  5. oEvent.addEvent = addEvent;
  6. oEvent.removeEvent = removeEvent;
  7. })();

示例四:

  1. var rainman = (function(x , y){
  2. return x + y;
  3. })(2 , 3);
  4. /**
  5. * 也可以写成下面的形式,因为第一个括号只是帮助我们阅读,但是不推荐使用下面这种书写格式。
  6. * var rainman = function(x , y){
  7. * return x + y;
  8. * }(2 , 3);
  9. */

在这里我们创建了一个变量rainman,并通过直接调用匿名函数初始化为5,这种小技巧有时十分实用。

示例五:

  1. var outer = null;
  2. (function(){
  3. var one = 1;
  4. function inner (){
  5. one += 1;
  6. alert(one);
  7. }
  8. outer = inner;
  9. })();
  10. outer(); //2
  11. outer(); //3
  12. outer(); //4

这段代码中的变量one是一个局部变量(因为它被定义在一个函数之内),因此外部是不可以访问的。但是这里我们创建了inner函数,inner函数是可以访问变量one的;又将全局变量outer引用了inner,所以三次调用outer会弹出递增的结果。

注意:闭包允许内层函数引用父函数中的变量,但是该变量是最终值

示例六:

  1. /**
  2. * <body>
  3. * <ul>
  4. * <li>one</li>
  5. * <li>two</li>
  6. * <li>three</li>
  7. * <li>one</li>
  8. * </ul>
  9. */
  10. var lists = document.getElementsByTagName('li');
  11. for(var i = 0 , len = lists.length ; i < len ; i++){
  12. lists[ i ].onmouseover = function(){
  13. alert(i);
  14. };
  15. }

你会发现当鼠标移过每一个<li&rt;元素时,总是弹出4,而不是我们期待的元素下标。这是为什么呢?注意事项里已经讲了(最终值)。显然这种解释过于简单,当mouseover事件调用监听函数时,首先在匿名函数( function(){ alert(i); })内部查找是否定义了 i,结果是没有定义;因此它会向上查找,查找结果是已经定义了,并且i的值是4(循环后的i值);所以,最终每次弹出的都是4。

解决方法一:

  1. var lists = document.getElementsByTagName('li');
  2. for(var i = 0 , len = lists.length ; i < len ; i++){
  3. (function(index){
  4. lists[ index ].onmouseover = function(){
  5. alert(index);
  6. };
  7. })(i);
  8. }

解决方法二:

  1. var lists = document.getElementsByTagName('li');
  2. for(var i = 0, len = lists.length; i < len; i++){
  3. lists[ i ].$$index = i; //通过在Dom元素上绑定$$index属性记录下标
  4. lists[ i ].onmouseover = function(){
  5. alert(this.$$index);
  6. };
  7. }

解决方法三:

  1. function eventListener(list, index){
  2. list.onmouseover = function(){
  3. alert(index);
  4. };
  5. }
  6. var lists = document.getElementsByTagName('li');
  7. for(var i = 0 , len = lists.length ; i < len ; i++){
  8. eventListener(lists[ i ] , i);
  9. }

以上就是详谈JavaScript函数闭包使用及注意事项代码详解的详细内容,更多请关注Gxl网其它相关文章!

人气教程排行