Javascript的闭包详解

在本人的博客:http://blog.csdn.net/u011043843/article/details/26148265  中也有对闭包的解释


在javascript中闭包是一个很不好理解的概念,但是确实一个不可逃避的东西,那么今天我们就来一起学习一下闭包。          

        什么是闭包?

        闭包:官方的解释是:闭包是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分。相信读完这句话以后,你就更加不知道什么是闭包了。其实通俗的说闭包就是一个函数a内部的局部变量s,被该函数内部的函数b所使用,并且a函数返回值为b函数。那么我们就将b函数成为闭包。


        为什么会产生闭包这个概念呢?那就要谈谈变量作用域的问题了。

        变量的作用域无非就是两种:全局变量和局部变量。

        Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量。


        Js代码

[javascript] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:18px;">var n=999;  
  2. function f1(){  
  3.     alert(n);  
  4. }  
  5. f1(); // 弹出对话框:999</span>  

        另一方面,在函数外部自然无法读取函数内的局部变量。

        Js代码

[javascript] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. functionf1(){  
  2.     varn=999;  
  3. }  
  4. alert(n); //弹出对话框:error  

         这里有一个地方需要注意,函数内部声明变量的时候,一定要使用var命令。如果不用的话,你实际上声明了一个全局变量!

         Js代码

[javascript] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:18px;">functionf1(){  
  2.    n=999;  
  3. }  
  4. f1();  
  5. alert(n); //999</span>  

         那么如何从外部读取局部变量?

         出于种种原因,我们有时候需要得到函数内的局部变量。但是,前面已经说过了,正常情况下,这是办不到的,只有通过变通方法才能实现。那就是在函数的内部,再定义一个函数。

        Js代码

[javascript] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:18px;">functionf1(){  
  2.     varn=999;  
  3.     functionf2(){  
  4.         alert(n); // 999  
  5.     }  
  6. }</span>  

        在上面的代码中,函数f2就被包括在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1 就是不可见的。这就是Javascript语言特有的“链式作用域”结构(chainscope),子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。

       既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!

       Js代码

[javascript] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:18px;">functionf1(){  
  2.     varn=999;  
  3.     functionf2(){  
  4.        alert(n);  
  5.     }  
  6.     returnf2;  
  7. }  
  8. varresult=f1();  
  9. result(); //999</span>  

       这个我们在函数函数f1的外部就可以读取到f1内的变量n的值了。

       大家可能注意到了,这个函数函数跟我上边的描述好像很吻合,没错,这就是闭包了。

       那么总结一下闭包都具备哪些特点呢?

       1,闭包外层是个函数.
       2,闭包内部都有函数.
       3,闭包会return内部函数.
       4,闭包返回的函数内部不能有return.(因为这样就真的结束了)


       闭包有什么作用呢?

       一个是像上边所说的那样,在函数外边访问函数内部的变量。另一个就是让这些变量的值始终保持在内存中。怎么理解他的第二个作用呢?

      看一下下边这个例子:

[javascript] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:18px;">Js代码  
  2. functionf1(){  
  3.     varn=999;  
  4.     functionf2(){  
  5.         alert(++n);  
  6.     }  
  7.     returnf2;  
  8. }  
  9. varresult=f1();  
  10. result(); // 999  
  11. result(); //1000</span>  

        大家可以看到两次执行同一个函数,结果却是不一样的,这个是为什么呢?为什么不像其他语言那个,一个函数执行完以后就被垃圾机制回收呢? 原因就在javascript的垃圾回收机制中,在Javascript中,如果一个对象不再被引用,那么这个对象就会被GC回收。如果两个对象互相引用,而不再被第3者所引用,那么这两个互相引用的对象也会被回收。因为函数f1被f2引用,f2又被f1外的c引用,这就是为什么函数f1执行后不会被回收的原因。


         使用闭包函数应该注意的问题;

         1)由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。

         2)闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(PublicMethod),把内部变量当作它的私有属性(privatevalue),这时一定要小心,不要随便改变父函数内部变量的值。


         闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现.所以学号闭包我们通往高级js程序员的一个必由之路。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值