定义
闭包是指有权访问另一个函数作用域中的变量的函数。
当在函数内部定义了其他函数时,就创建了闭包。要理解闭包,就要理解变量的作用域,闭包的原理就是作用域的访问原则。
函数内部可以访问外部变量和自身,但外部无法对其函数内部进行访问。但有时我们需要得到内部的变量,怎么办?
在函数内部再定义一个函数。
function f1(){
var a=123;
function f2(){
alert(a); // 123
}
}
函数f2就被包括在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1就是不可见的。既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!
function f1(){
var a=123;
function f2(){
alert(n);
}
return f2;
}
var result=f1();
result(); // 123
通常,函数的作用域及其所有变量都会在函数执行结束后被销毁,但当函数返回了一个闭包时,这个函数的作用域将会一直在内存中保存到闭包不存在为止。因此闭包的缺点是会常驻内存,会增大内存使用量,使用不当很容易造成内存泄漏(是指程序在申请内存后,无法释放已申请的内存空间)。
关于this对象
在全局函数中,this等于window,而当函数被作为某个对象的方法调用时,this等于那个对象。不过,匿名函数的执行环境具有全局性,因此其this对象通常指向window。
var name = "The Window";
var object = {
name : "My Object",
getNameFunc : function(){
return function(){
return this.name;
};
}
};
alert(object.getNameFunc()());//The Window
但把外部作用域中的this对象保存在一个闭包能够访问到的变量里,就可以让闭包访问该对象了。
var name = "The Window";
var object = {
name : "My Object",
getNameFunc : function(){
var that = this;
return function(){
return that.name;
};
}
};
alert(object.getNameFunc()());//My Object
在定义匿名函数之前,我们把this对象赋值给了一个名叫that的变量,而在定义了闭包之后,闭包也可以访问这个变量,因为它是我们在包含函数中特意声明的一个变量。即使在函数返回之后,that也仍然引用着object,所以调用object,所以调用object.getNameFunc()()就返回了“My Object”。