首先要区分两个概念,一是匿名函数,一是闭包。
所谓匿名函数,就是创建函数没有给定函数名。经常出现的包括函数表达式,就是定义一个匿名函数,然后将函数赋值给某个变量,而此时这个变量就相当于该函数的函数名,例如:
var sayHi = function(){ alert("Hi"); }; //注意这个分号 sayHi(); //调用函数
还有一种常用匿名函数的情况是回调函数,如 JQuery 中常用到的:
$("p").click(function(){ alert("click"); });
此外,还有利用匿名函数作为某函数的返回值:
function sayNameWithAge(age){ return function(person){ if(person.age == age){ return person.name; } } }
那么,闭包又是怎么一回事呢?所谓的闭包,其实就是一个函数,而这个函数有一点比较特别,它有权能够去访问其他函数作用域的变量。
从定义中我们发现,其实在上面的匿名函数例子中,就存在这样的闭包。在最后一个例子中,匿名函数访问了函数 sayNameWithAge 的参数 age,那么,这个作为返回值的匿名函数就是一个闭包。
要彻底理解闭包,就必须理解函数调用时的整个机制,这里从作用域链的相关知识来进行讲解。
首先看下面的例子:
function sayName(name){ alert(name); } sayName("Jack");
在上面的函数 sayName 被调用的时候,就会创建一个对应的执行环境和作用域链,如下图所示:
当 sayName 函数被调用时,创建了相应的作用域链,而作用域中包含两个引用分别指向两个对象,其中一个是全局变量对象,这个全局对象是在函数创建的时候就已经创建了,只是在调用函数的时候才将其复制到作用域链中;而另一个就是函数的活动对象,这个对象是在调用函数的时候才创建的。
在函数中访问一个变量时,就会从作用域中搜索对应名字的变量。
而当函数执行完毕后,函数的活动对象会被销毁,而全局变量对象却永远保存在内存中。
但是,上面所说的都是普通函数的情况,对于闭包而言,又是另外一种情况:
以上面的 sayNameWithAge 函数为例:
function sayNameWithAge(age){ return function(person){ if(person.age == age){ return person.name; } } } //创建函数 var sayName = sayNameWithAge(18); //调用函数 var name = sayName({name:"Jack",age:18}); //解除对匿名函数的引用 sayName = null;
当上面的 sayName 函数被调用的时候,产生的作用域链如下所示:
当匿名函数被 return 后,它的作用域链被创建,并且包含了外部函数的活动对象和全局变量对象,这样一来,这个匿名函数就可以访问 sayNameWithAge 函数中定义的所有变量,也就是一个闭包。
这样的闭包会存在一个问题,就是当 sayNameWithAge 函数执行完毕的时候(JS 的垃圾处理机制大多是标记清除),其活动对象被闭包所引用,所以活动对象并不会被销毁,只有当匿名函数被销毁后,sayNameWithAge 的活动对象才会被销毁,所以上面的最后一行解除对匿名函数的引用不仅是为了销毁闭包的对象,也是为了销毁外部函数的活动对象。所以,慎重使用闭包!!!
关于闭包,还有一个需要注意的地方,就是在闭包中访问其他函数的变量,实际上是因为闭包的作用域链中有指向其他函数的活动对象的引用,而不是闭包自身的活动对象中保存着这些变量。看下面的例子:
function outer(){ var result = new Array(); for(var i = 0; i < 5; i ++){ result[i] = function(){ return i; }; } return result; }
按照设想,最后 outer 返回的数组各个项中的值应该是与其下标一致的。但是,最后的结果却是每个项的值都是 5
不难想象,在上面的所有闭包的作用域链中,都有一个引用指向了 outer 的活动对象中的参数 i,而且是指向同一个对象。
当 outer 函数执行完毕的时候,i 的值是 5。也就是说,所有闭包中访问 i 的时候取到的值都是 5
那么,我们可以通过另一种方法来实现预想的效果:
function outer(){ var result = new Array(); for(var i = 0; i < 5; i ++){ result[i] = (fuction(index){ return index; })(i); } return result; }
这里我们为匿名函数定义一个参数 index,并在每次循环中立即调用该函数,将 i 的当前值复制给参数 index(注意 JS 中是按值传递),并将返回的 index 赋值给 result。
此外,闭包中需要注意的另一个问题是 this 对象。
this 对象在 JS 中是在函数运行时基于函数的执行环境绑定的。而匿名函数的执行环境具有全局性,也就是说,在匿名函数中,this 对象通常指向 window。
var name = "Tom"; var person = { name : "Jack", sayName : function(){ return (function(){ return this.name; })(); } } person.sayName(); //Tom
上面在闭包中访问 this.name,其中的 this 对象并非取得自身或是 person 的 this 对象,而是指向 window。
如果需要在闭包中访问外部函数的 this 对象,那么,可以在外部函数中定义一个变量,将 this 对象传给该变量。
var name = "Tom"; var person = { name : "Jack", sayName : function(){ var self = this; return (function(){ return self.name; })(); } } person.sayName(); //Jack
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
本文向大家介绍Linux INotif机制详解及实例代码,包括了Linux INotif机制详解及实例代码的使用技巧和注意事项,需要的朋友参考一下 Linux INotif机制 一、 前言: 众所周知,Linux 桌面系统与 MAC 或 Windows 相比有许多不如人意的地方,为了改善这种状况,开源社区提出用户态需要内核提供一些机制,以便用户态能够及时地得知内核或底层硬件设备发生了什么,从而能够
本文向大家介绍javascript 闭包详解及简单实例应用,包括了javascript 闭包详解及简单实例应用的使用技巧和注意事项,需要的朋友参考一下 JS 闭包详解及实例: 最近学习JS的基础知识,学习了闭包的知识点,有很多疑惑,这一段时间还是一直有在看闭包的相关知识理解就更深入了一点,下面说说我的理解。 如上所示,上面第一个return返回的就是一个闭包,那么本质上说闭包就是一个函数。那么返回
本文向大家介绍C++ 反射机制详解及实例代码,包括了C++ 反射机制详解及实例代码的使用技巧和注意事项,需要的朋友参考一下 C++ 反射机制 一.前言: Java有着一个非常突出的动态相关机制:Reflection,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括meth
本文向大家介绍Java 反射机制详解及实例代码,包括了Java 反射机制详解及实例代码的使用技巧和注意事项,需要的朋友参考一下 Java反射详解 本篇文章依旧采用小例子来说明,因为我始终觉的,案例驱动是最好的,要不然只看理论的话,看了也不懂,不过建议大家在看完文章之后,在回过头去看看理论,会有更好的理解。 下面开始正文。 【案例1】通过一个对象获得完整的包名和类名 【运行结果】:Reflect.D
本文向大家介绍Java 回调机制(CallBack) 详解及实例代码,包括了Java 回调机制(CallBack) 详解及实例代码的使用技巧和注意事项,需要的朋友参考一下 Java 回调机制 概要: 最近学习java,接触到了回调机制(CallBack)。初识时感觉比较混乱,而且在网上搜索到的相关的讲解,要么一言带过,要么说的比较单纯的像是给CallBack做了一个定义。当然了,我在理解了回调之
本文向大家介绍JavaScript 栈的详解及实例代码,包括了JavaScript 栈的详解及实例代码的使用技巧和注意事项,需要的朋友参考一下 JavaScript 栈 栈是一种遵从先进后出(LIFO)原则的有序集合。 新添加或待删除的元素都保存在栈的末尾,称作栈顶,另一端就叫栈底。 在栈里,新元素都靠近栈顶,旧元素都接近栈底 昨天因为有点事没有更新,今天打算给大家讲讲JavaScript实现的数