Javascript和Java中闭包的理解

一。Javascript中闭包:

1.变量的作用域

  要理解闭包,首先必须理解Javascript特殊的变量作用域。
  变量的作用域无非就是两种:全局变量和局部变量。
  
  Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量 
 var n=999;
 function f1(){
   alert(n);
 }
  f1(); // 999

  另一方面,在函数外部自然无法读取函数内的局部变量。 
 function f1(){
   var n=999;
 }
  alert(n); // error

  这里有一个地方需要注意,函数内部声明变量的时候,一定要使用var命令。如果不用的话,你实际上声明了一个全局变量! 
 function f1(){
   n=999;
 }
  f1();
  alert(n); // 999

2.如何从外部读取局部变量

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

  既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!
  function f1(){
    n=999;
    function f2(){
      alert(n);
   }
   return f2;
 }
 var result=f1();
 result(); // 999

3.闭包的概念

  上面的f2函数,就是闭包。闭包其实就是定义在一个函数内部的函数(因为是子函数所以能够读取所在父函数的内部变量)。
  在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。

4.闭包的用途

  1.可以读取函数内部的变量.
  2.让这些变量(闭包引用的变量)的值始终保持在内存中。
  function f1(){
    var n=999;
    nAdd=function(){n+=1}
    function f2(){
      alert(n);
    }
    return f2;
 }
 var result=f1();
 result(); // 999
 nAdd();
 result(); // 1000
        在这段代码中,result实际上就是闭包f2函数。它一共运行了两次,第一次的值是999,第二次的值是1000。这证明了,函数f1中的局部变量n一直保存在内存中,并没有在f1调用后被自动清除。
        为什么会这样呢?原因就在于f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依赖于f1,因此f1也始终在内存中,不会在调用结束后,被垃圾回收机制(garbage collection)回收。
        这段代码中另一个值得注意的地方,就是“nAdd=function(){n+=1}”这一行,首先在nAdd前面没有使用var关键字,因此 nAdd是一个全局变量,而不是局部变量。其次,nAdd的值是一个匿名函数(anonymous function),而这个

       匿名函数本身也是一个闭包,所以nAdd相当于是一个setter,可以在函数外部对函数内部的局部变量进行操作。


5.使用闭包的注意点

  1.由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包

  2.闭包会在父函数外部,改变父函数内部变量的值。


6.闭包的应用场景

  1、保护函数内的变量安全。以最开始的例子为例,函数a中i只有函数b才能访问,而无法通过其他途径访问到,因此保护了i的安全性。
  2、在内存中维持一个变量。依然如前例,由于闭包,函数a中i的一直存在于内存中,因此每次执行c(),都会给i自加1。
 

二。Java中的闭包

1.java中闭包的定义

  定义:闭包能够将一个方法作为一个变量去存储,这个方法有能力去访问所在类的自由变量。

  Java 语言本身还没有正式支持闭包,但它却允许模拟闭包(内部类+接口)。可以使用匿名的内部类来实现闭包。

  如何让这个普通对象能够访问所在类的自由变量?内部类。内部类能够访问外部类的所有属性及方法。

  隐藏具体实现是内部类的作用之一,如何保证隐藏具体实现的同时还能将闭包传递到外部使用?让内部类实现通用接口,然后将内部类对象向上转型为接口类型。

2.代码简单实现:

  	public final static String name = "纯牛奶";// 名称


		private static int num = 16;// 数量


		public Milk() {
			System.out.println(name + ":16/每箱");
		}


		/**
		 * 闭包
		 * 
		 * @return 返回一个喝牛奶的动作
		 */
		public Active HaveMeals() {
			return new Active() {
				public void drink() {
					if (num == 0) {
						System.out.println("木有了,都被你丫喝完了.");
						return;
					}
					num--;
					System.out.println("喝掉一瓶牛奶");
				}
			};
		}


		/**
		 * 获取剩余数量
		 */
		public void currentNum() {
			System.out.println(name + "剩余:" + num);
		}
	}


	/**
	 * 通用接口
	 */
	interface Active {
		void drink();
	}  
	
	//闭包的使用
	public class Person {   
		public static void main(String[] args) {  
			//买一箱牛奶  
			Milk m = new Milk(); 
			Active haveMeals = m.HaveMeals();
			//没事喝一瓶  
			haveMeals.drink();  
			//有事喝一瓶  
			haveMeals.drink(); 
			//看看还剩多少?  
			m.currentNum();  
		}   
	}  

4.总结: 

   1.实际项目中没怎么用过闭包,因此不能对他的好坏进行评论。 
   2.建议合理的使用闭包,不完全不使用,也不能滥用。 
   3.特别注意:闭包会导致资源不被回收,如上例,在main方法中将m设为null,使用haveMeals继续调用drink方法仍然会喝掉一瓶牛奶,说明Milk对象并没有被释放掉。

  
  

你可能感兴趣的:(原理解析)