闭包

闭包是指有权访问另一个函数作用域中的变量的函数。创建闭包的常见方式,就是在一个函数内部创建另一个函数。

function fn(propertyName){

         return function(object1){

                   var v1 = object1[propertyName]; //访问了外部变量propertyName变量

         }

}

由于闭包会携带包含它的函数的作用域,因此会比其他函数占用更多的内存,过度使用闭包可能会导致内存占用过多。所以建议只是在绝对必要时再考虑使用闭包。


function createFunctions(){
     var result = [];
     for(var i=0; i<10; i++){
           result[i] = (function(num){
                return num;
           })(i);
     }
     return result
}
var funcs = createFunctions();
for(var i=0; i<funcs.length; i++){
    document.write(funcs[i] + '<br />');
}

好像挺简单.但是闭包到底有什么作用呢?看一个例子。

我们想写一个方法,每次都得到一个整数,这个整数是每次加1的,没有思索,马上下笔:

Js代码   收藏代码
  1. var i = 0;  
  2. function getNext() {  
  3.     i++;  
  4.     return i;  
  5. }  
  6. alert(getNext()); //1  
  7. alert(getNext()); //2  
  8. alert(getNext()); //3  

 

一直用getNext函数得到下一个整数,而后不小心或者故意的将全局变量i的值设为0,然后再次调用getNext,
你会发现又从1开始了........这时你会想到,要是把i设置成一个私有变量该多好,这样只有在方法内部才
可能改变它,在函数之外就没有办法修改了.下面的代码就是按照这个要求来做得,后面我们详细讨论。
为了解释方便,我们就把下面的代码称为demo1.

Js代码   收藏代码
  1. function temp() {  
  2.     var i = 0;  
  3.     function b() {  
  4.         return ++i;  
  5.     }  
  6.     return b;  
  7. }  
  8. var getNext = temp();  
  9. alert(getNext());    //1  
  10. alert(getNext());    //2  
  11. alert(getNext());    //3  
  12. alert(getNext());    //4  

 

因为我们平时所说的javascript绝大多数都是指的在客户端(浏览器)下,所以这里也不例外。
在javascript解释器启动时,会首先创建一个全局的对象(global object),也就是"window"所引用的对象.
然后我们定义的所有全局属性和方法等都会成为这个对象的属性.
不同的函数和变量的作用域是不同的,因而构成了一个作用域链(scope chain).很显然,在javascript解释器启动时,
这个作用域链只有一个对象:window(Window Object,即global object).
在demo1中,temp函数是一个全局函数,因此temp()函数的作用域(scopr)对应的作用域链就是js解释器启动时的作用域链,只有一个window对象。
当temp执行时,首先创建一个call对象(活动对象),然后把这个call对象添加到temp函数对应的作用域链的最前头,这是,temp()函数
对应的作用域链就包含了两个对象:window对象和temp函数对应的call object(活动对象).然后呢,因为我们在temp函数里定义了变量i,
定义了函数b(),这些都会成为call object的属性。当然,在这之前会首先给call object对象添加arguments属性,保存了temp()函数执行时
传递过来的参数。此时,整个的作用域链如下图所示:


同理可以得出函数b()执行时的整个作用域链:


注意在b()的作用域链中,b()函数对应的call object只有一个arguemnts属性,并没有i属性,这是因为在b()的定义中,并没有用var关键字来
声明i属性,只有用var 关键字声明的属性才会添加到对应的call object上.
在函数执行时,首先查找对应的call object有没有需要的属性,如果没有,再往上一级查找,直到找到为止,如果找不到,那就是undefined了.

这样我们再来看demo1的执行情况。我们用getNext引用了temp函数,而temp函数返回了函数b,这样getNext函数其实就是b函数的引用。
执行一次getNext,就执行一次b()函数。因为函数b()的作用域依赖于函数temp,因此temp函数在内存中会一直存在。函数b执行时,首先查找
i,在b对应的call object中没有,于是往上一级找,在temp函数对应的call object中找到了,于是将其值加1,然后返回这个值。
这样,只要getNext函数有效,那么b()函数就一直有效,同时,b()函数依赖的temp函数也不会消失,变量i也不会消失,而且这个变量在temp函数
外部根本就访问不到,只能在temp()函数内部访问(b当然可以了).

来看一个利用闭包来模拟私有属性的例子:

Js代码   收藏代码
  1. function Person(name, age) {    
  2.     this.getName = function() { return name; };    
  3.     this.setName = function(newName) { name = newName; };    
  4.     this.getAge = function() { return age; };    
  5.     this.setAge = function(newAge) { age = newAge; };    
  6. }    
  7.     
  8. var p1 = new Person("sdcyst",3);    
  9. alert(p1.getName());  //sdcyst    
  10. alert(p1.name);       //undefined   因为Person('类')没有name属性    
  11. p1.name = "mypara"    //显示的给p1添加name属性    
  12. alert(p1.getName());  //sdcyst     但是并不会改变getName方法的返回值    
  13. alert(p1.name);       //mypara     显示出p1对象的name属性    
  14. p1.setName("sss");    //改变私有的"name"属性  
  15. alert(p1.getName());  //sss    
  16. alert(p1.name);       //仍旧为mypara    

 

定义了一个Person类,有两个私有属性name,age,分别定义对应的get/set方法。
虽然可以显示的设置p1的name、age属性,但是这种显示的设置,并不会改变我们
最初设计时模拟出来的"name/age"私有属性。

解释闭包的确不是一件容易的事,在网上很多人也是利用例子来说明闭包。如果有地方说的不对,还请指正。
下面是另一篇解释javascript闭包的文章,一块儿参考吧。
http://softbbs.pconline.com.cn/9497825.html


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值