学习闭包
何为闭包?##
闭包是指 有权访问 另一个函数作用域 中的变量 的函数,常见的创建方式是 **在一个函数的内部创建另一个函数 **,通过另一个函数访问这个函数的局部变量。
闭包的特性##
闭包有三个特性:
- 函数嵌套函数
- 函数内部可以引用外部的参数和变量
- 参数和变量不会被垃圾回收机制回收
闭包是javascript
语言的一大特点,主要应用闭包场合主要是为了:设计私有的方法和变量。
闭包的缺点就是常驻内存,会增大内存使用量,使用不当很容易造成内存泄露。
一般函数执行完毕后,局部活动对象就被销毁,内存中仅仅保存全局作用域。但闭包的情况不同!
javascript
function aaa() {
var a = 1;
return function(){
alert(a++)
};
}
var fun = aaa();
fun();// 1 执行后 a++,,然后a还在~
fun();// 2
fun = null;//a被回收!!
闭包会使变量始终保存在内存中,如果不当使用会增大内存消耗。
javascript的垃圾回收原理##
- 在javascript中,如果一个对象不再被引用,那么这个对象就会被GC回收;
- 如果两个对象互相引用,而不再被第3者所引用,那么这两个互相引用的对象也会被回收。
使用闭包的好处##
那么使用闭包有什么好处呢?使用闭包的好处是:
1. 希望一个变量长期驻扎在内存中
2. 避免全局变量的污染
3. 私有成员的存在
全局变量的累加###
javascript
<script>
var a = 1;
function abc(){
a++;
alert(a);
}
abc(); //2
abc(); //3
</script>
局部变量###
javascript
<script>
function abc(){
var a = 1;
a++;
alert(a);
}
abc(); //2
abc(); //2
</script>
那么怎么才能做到变量a既是局部变量又可以累加呢?
局部变量的累加###
javascript
<script>
function outer(){
var x=10;
return function(){ //函数嵌套函数
x++;
alert(x);
}
}
var y = outer(); //外部函数赋给变量y;
y(); //y函数调用一次,结果为11,相当于outer()();
y(); //y函数调用第二次,结果为12,实现了累加
</script>
函数声明与函数表达式###
定义函数有两种方式:一种是函数声明,另一种就是函数表达式
javascript
//1,函数声明
function functionName(arg0, arg1, arg2){
//函数体
}
//函数声明的一个重要特征是,函数声明提升,意思是**在执行代码之前**会先**读取函数声明**。而函数表达式没有这个特征。
eg: sayHi();
function sayHi(){
alert('Hi!');
}
//2,函数表达式
var functionName = function(arg0, arg1, arg2){
//函数体
}
在JS中函数可以分为两种,具名函数(命名函数)和匿名函数。
区分这两种函数的方法非常简单,可以通过输出fn.name
来判断,有name
的就是具名函数,没有`name的就是匿名函数
注意:在低版本IE上无法获取具名函数的name,会返回undefined
,建议在火狐或是谷歌浏览器上测试
或是采用兼容IE的获取函数name方法来获取函数名称:
javascript
/**
* 获取指定函数的函数名称(用于兼容IE)
* @param {Function} fun 任意函数
*/
function getFunctionName(fun) {
if (fun.name !== undefined)
return fun.name;
var ret = fun.toString();
ret = ret.substr('function '.length);
ret = ret.substr(0, ret.indexOf('('));
return ret;
}
在js中我们可以通过关键字function来声明一个函数:
javascript
<script>
function abc(){
alert(123);
}
abc();
</script>
我们也可以通过一个"()"来将这个声明变成一个表达式:
javascript
<script>
(function (){
alert(123);
})(); //然后通过()直接调用前面的表达式即可,因此函数可以不必写名字;
</script>
模块化代码,减少全局变量的污染###
javascript
<script>
var abc = (function(){ //abc为外部匿名函数的返回值
var a = 1;
return function(){
a++;
alert(a);
}
})();
abc(); //2 ;调用一次abc函数,其实是调用里面内部函数的返回值
abc(); //3
</script>
私有成员的存在###
javascript
<script>
var aaa = (function(){
var a = 1;
function bbb(){
a++;
alert(a);
}
function ccc(){
a++;
alert(a);
}
return {
b:bbb, //json结构
c:ccc
}
})();
aaa.b(); //2
aaa.c() //3
</script>
使用匿名函数实现累加###
javascript
//使用匿名函数实现局部变量驻留内存中,从而实现累加
<script type="text/javascript">
function box(){
var age = 100;
return function(){ //匿名函数
age++;
return age;
};
}
var b = box();
alert(b());
alert(b()); //即alert(box()());
alert(b());
alert(b); // function () {
// age++;
// return age;
// }
b = null; //解除引用,等待垃圾回收
</script>
过度使用闭包会导致性能的下降。函数里放匿名函数,则产生了闭包
在循环中直接找到对应元素的索引###
html
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />
<title></title>
<script>
window.onload = function(){
var aLi = document.getElementsByTagName('li');
for (var i=0;i<aLi.length;i++){
aLi[i].onclick = function(){ //当点击时for循环已经结束
alert(i);
};
}
}
</script>
</head>
<body>
<ul>
<li>123</li>
<li>456</li>
<li>789</li>
<li>010</li>
</ul>
</body>
</html>
使用闭包改写上面代码###
css
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />
<title></title>
<script>
window.onload = function(){
var aLi = document.getElementsByTagName('li');
for (var i=0;i<aLi.length;i++){
(function(i){
aLi[i].onclick = function(){
alert(i);
};
})(i);
}
};
</script>
</head>
<body>
<ul>
<li>123</li>
<li>456</li>
<li>789</li>
</ul>
</body>
</html>
内存泄露问题###
由于IE的js对象和DOM对象使用不同的垃圾收集方法,因此闭包在IE中会导致内存泄露问题,也就是无法销毁驻留在内存中的元素
javascript
function closure(){
var oDiv = document.getElementById('oDiv');//oDiv用完之后一直驻留在内存中
oDiv.onclick = function () {
alert('oDiv.innerHTML');//这里用oDiv导致内存泄露
};
}
closure();
//最后应将oDiv解除引用来避免内存泄露
function closure(){
var oDiv = document.getElementById('oDiv');
var test = oDiv.innerHTML;
oDiv.onclick = function () {
alert(test);
};
oDiv = null;
}
10.新增
很多初学者学了闭包后,可能也不懂到底在哪里才有用到,甚至不懂闭包到底有啥用。其实你写的每一个js函数都是闭包,一个js函数的顶层作用域就是window对象,js的执行环境本身就是一个scope(浏览器的window/node的global),我们通常称之为全局作用域。每个函数,不论多深,都可以认为是全局scope的子作用域,可以理解为闭包。