闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。
闭包的特性
1.函数嵌套函数
2.函数内部可以引用外部的参数和变量
3.参数和变量不会被垃圾回收机制回收
闭包的定义及其优缺点
闭包 是指有权访问另一个函数作用域中的变量的函数,创建闭包的最常见的方式就是在一个函数内创建另一个函数,通过另一个函数访问这个函数的局部变量
闭包的缺点就是常驻内存,会增大内存使用量,使用不当很容易造成内存泄露。
闭包是javascript语言的一大特点,主要应用闭包场合主要是为了:设计私有的方法和变量。
一般函数执行完毕后,局部活动对象就被销毁,内存中仅仅保存全局作用域。但闭包的情况不同!
嵌套函数的闭包
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 
 | function aaa() {
 
 var a = 1
 return function() {
 
 alert(a++)
 
 }
 
 }
 
 var fun = aaa();
 
 fun();// 1 执行后 a++,,然后a还在~
 
 fun();// 2
 
 fun = null;//a被回收!!
 
 
 | 
闭包会使变量始终保存在内存中,如果不当使用会增大内存消耗。
javascript的垃圾回收原理
1.在javascript中,如果一个对象不再被引用,那么这个对象就会被GC回收。
2.如果两个对象互相引用,而不再被第3者所引用,那么这两个互相引用的对象也会被回收。
使用闭包的好处
1.希望一个变量长期驻扎在内存中
2.避免全局变量的污染
3.私有成员的存在
1.全局变量的累加
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 
 | var a = 1;
 
 function abc(){
 
 a++;
 
 alert(a);
 
 }
 
 abc(); //2
 
 abc(); //3
 
 
 | 
2.局部变量
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 
 | function abc(){
 
 var a = 1;
 
 a++;
 
 alert(a);
 
 }
 
 abc(); //2
 
 abc(); //2
 
 
 | 
3.局部变量的累加
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 
 | function outer(){
 
 var x=10;
 
 return function(){ //函数嵌套函数
 
 x++;
 
 alert(x);
 
 }
 
 }
 
 var y = outer(); //外部函数赋给变量y;
 
 y(); //y函数调用一次,结果为11,相当于outer()();
 
 y(); //y函数调用第二次,结果为12,实现了累加
 
 
 | 
4.模块化代码,减少全局变量的污染
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 
 | var abc = (function(){ //abc为外部匿名函数的返回值
 
 var a = 1;
 
 return function(){
 
 a++;
 
 alert(a);
 
 }
 
 })();
 
 abc();    //2 ;调用一次abc函数,其实是调用里面内部函数的返回值
 
 abc();    //3
 
 
 | 
5.私有变量的存在
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 
 | var aaa = (function(){
 
 var a = 1;
 
 function bbb(){
 
 a++;
 
 alert(a);
 
 }
 
 function ccc(){
 
 a--;
 
 alert(a);
 
 }
 
 return {
 
 b:bbb,
 
 c:ccc
 
 }
 
 })();
 
 aaa.b();     //2
 
 aaa.c()      //1
 
 
 | 
6.使用匿名函数实现累加
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 
 | function box(){
 
 var age = 100;
 
 return function(){          //匿名函数
 
 age++;
 
 return age;
 
 }
 
 }
 
 var b = box();
 
 alert(b()); //即alert(box()());
 
 alert(b);
 
 b = null; //解除引用,等待垃圾回收
 
 
 | 
7.在循环中添加点击事件
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 
 | window.onload = function(){
 
 var aLi = document.getElementsByTagName('li');
 
 for (var i=0;i<aLi.length;i++){
 
 aLi[i].onclick = function(){ //当点击时for循环已经结束
 
 alert(i);
 
 }
 
 }
 
 }
 
 
 | 
8.使用闭包改写上面代码
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 
 | 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)
 
 }
 
 }
 
 
 | 
9.内存泄露问题
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 
 | 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;
 
 }
 
 
 |