在JS这块,免不了被问什么是闭包。

  从一个常见的循环问题说起。

  有一个ul列表, 里面有5个li标签,我希望点击每个li标签的时候,弹出每个li标签对应的索引值(第一个弹出0,第二个弹出1...)。

<ul id="result">
<li>1</li>
<li>2</li>
<li>3</li>
<li>4</li>
<li>5</li>
</ul>

  当我很认真的写出一段代码:

var lis = document.getElementsByTagName('li'), n = lis.length, i = 0;
for(; i < n; i++){
lis[i].onclick = function(){
alert(i);
}
}

  蛮高兴的做了点击测试,从第一个li标签开始,弹出"5",第二个、第三个...全部都弹出“5”。什么情况,这不是我想要的结果。出现了问题,就去找问题的原因,当我点击每个li标签的时候,都弹出“5”,说明for循环已经运行完了,变量 也跟着循环条件增加到了5,在我点击前,循环已经执行完成。

  经过一番思考,重新写了这段代码:

var lis = document.getElementsByTagName('li'), n = lis.length, i = 0;
for(; i < n; i++){
lis[i].index = i;
lis[i].onclick = function(){
alert(this.index);
}
}

  把每次循环时变量 i 的值赋值给每个li标签对象的一个属性。很神奇的这段代码做到了我要的结果,点击每个li标签弹出了对应的索引值(第一个弹出0,第二个弹出1...)。这让我想到是因为变量 i 没有被正确的引用,才发生那都弹出5的问题。懵懵懂懂的想到要正确的引用 变量 i 。经过多次写写改改,点击测试,写出了下面这样的代码:

var lis = document.getElementsByTagName('li'), n = lis.length, i = 0;
for(; i < n; i++){
(function(num){
lis[num].onclick = function(){
alert(num);
}
}(i));
}
//或者
var lis = document.getElementsByTagName('li'), n = lis.length, i = 0;
for(; i < n; i++){
lis[i].onclick = function(num){
return function(){
alert(num);
}
}(i);
}

  后来半知半解的明白了这是闭包的一种运用,闭包与变量的作用域变量的生存周期有密切的关系。要理解闭包就要理解变量的作用域、变量的生存周期。好吧,得先了解与变量有关的知识了。

  变量的作用域

  当在一个函数中声明一个变量的时候,如果我们没有加上关键字 var, 这个变量就是全局变量,加上了关键字var,这个变量就是局部变量,只有在这个函数内部才能访问这个局部变量,在函数外是访问不到的。 函数的参数也是局部变量,只能在函数内部访问。

function a(){
b = 1; //全局变量
var c = 2; //局部变量
}
a();
alert(b); //1
alert(c); //出错 c未定义

  定义了一个函数a,里面有个全局变量b,局部变量c。当在函数a外部访问变量b、c,正常弹出了b的值,而变量c是局部变量,没有正常访问,所以出错,提示变量c未定义。

  在函数内部,局部变量优先级高于同名的全局变量。当定义一个函数的时候,也会随之创建一个函数作用域。当在函数内部访问一个变量的时候,会在函数内部作用域搜索这个变量,如果函数内部没有这个变量,会在函数外部搜索,直到找到这个名称的变量为止。如果找不到,就会抛出一个错误。

var v = 1;
function a(){
var m = 2;
function b(){
var n = 3;
alert ( m ); // 2
alert ( v ); // 1
}
b();
alelrt ( n ); //出错
}
a();

       

  上面的代码第一次弹出变量m,首先在函数b里查找,但没有找到,继续往外找,在函数a里面找,m的值为2;第二次弹出变量v,同样的函数b里找,没有找到,再在函数a里找,也没有找到,在往外找,找到了v的值为1;第三次弹出变量n,在函数a里面找,没有找到,不能在函数b里面找,因为查找是向上、向外的,不能向下、向内,最后在函数a的外面找,也没有找到,这时就抛出错误,变量n没有定义。

  上面的代码有三个作用域,函数b的作用域,函数a的作用域,window全局作用域(最顶层的作用域),这些作用域联合起来,就形成了作用域链。访问变量就是在这个作用域链的一个搜索过程。

  变量的生存周期  

  在javascript中,全局变量拥有很长的生存周期,直到把变量销毁,而局部变量会随着函数调用结束被销毁。

function a(){
v = 1; //全局变量v
alert(v); //1
}
a();
alert(v) //1 全局变量v还存在 function b(){
var i = 2; //局部变量i
alert(i); //2
}
b();
alert(i) //出错 i未定义 局部变量i已经被销毁

  上面的代码定义了两个函数,函数a内部定义全局变量v,函数a执行完后全局变量v还存在;函数b内部定义了局部变量i,函数b执行完后局部变量i被销毁。

function c(){
var k = 3; //局部变量k
return function(){
k++;
alert(k);
}
}
var f = c();
f(); //4
f(); //5
f(); //6

  上面的代码定义了一个函数c,函数c内部定义了局部变量k,当函数c执行后返回了一个匿名函数,匿名函数访问了局部变量k,变量f的值为这个匿名函数,调用f实际上是调用这个匿名函数。每次调用f(),变量k的值都会增加1。在这里局部变量k没有在函数c执行完后被销毁,反而“活”了下来,它的生存周期延长了。

  什么是闭包

  当前作用域总是能够访问外部作用域中的变量,  函数是 JavaScript 中唯一拥有自身作用域的结构, 因此闭包的创建依赖于函数。

  1. 一个函数可以引用外部函数的变量,这个函数就可算是一个闭包。

  2. 外部函数已经执行完,内部的函数仍可以引用外部函数的变量。这个内部函数就可算是一个闭包。

  3. 函数能存储其作用域的变量、能读写当前函数作用域内变量的函数可算是一个闭包。

理解JavaScript的闭包的更多相关文章

  1. 深入理解JavaScript的闭包特性如何给循环中的对象添加事件

    初学者经常碰到的,即获取HTML元素集合,循环给元素添加事件.在事件响应函数中(event handler)获取对应的索引.但每次获取的都是最后一次循环的索引.原因是初学者并未理解JavaScript ...

  2. 深入理解javascript的闭包

    闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域 ...

  3. 如何给循环中的对象添加事件--深入理解JavaScript的闭包特性

    初学者经常碰到的,即获取HTML元素集合,循环给元素添加事件.在事件响应函数中(event handler)获取对应的索引.但每次获取的都是最后一次循环的索引.原因是初学者并未理解JavaScript ...

  4. 理解Javascript 的闭包(closure)

    要理解闭包的概念先从变量的作用域说去 一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之 ...

  5. 深入理解JavaScript的闭包特性 如何给循环中的对象添加事件(转载)

    原文参考:http://blog.csdn.net/gaoshanwudi/article/details/7355794 初学者经常碰到的,即获取HTML元素集合,循环给元素添加事件.在事件响应函数 ...

  6. 理解javascript的闭包,原型,和匿名函数及IIFE

    理解javascript的闭包,原型,和匿名函数(自己总结) 一 .>关于闭包 理解闭包 需要的知识1.变量的作用域 例1: var n =99; //建立函数外的全局变量 function r ...

  7. 简单理解javascript的闭包

    看过网上关于javascript的闭包的概念和分析,看完之后都是一头雾水,完全不懂,零度我本来就对于概念性的东西很烦躁,没办法,硬着头皮翻阅了很多的资料,总算理清了一点头绪,现在分享给大家,错误之处还 ...

  8. 【转】理解JavaScript之闭包

    闭包(closure)是掌握Javascript从人门到深入一个非常重要的门槛,它是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现.下面写下我的学习笔记~ 闭包-无处不 ...

  9. 理解 Javascript 的闭包

    什么是闭包 闭包是什么?闭包是Closure,这是静态语言所不具有的一个新特性.但是闭包也不是什么复杂到不可理解的东西,简而言之,闭包就是: 闭包就是函数的局部变量集合,只是这些局部变量在函数返回后会 ...

随机推荐

  1. Cocos2d-x串算出Size方法

    项目需要,根据所输入的字符串,的需要计算串帐户Size. 包代码如下面.只需要传递一个字符串,您可以返回Size: Size ChartDemoScene::calculateFontSize(con ...

  2. HTML5线性图表 图表数据区域可着色

    这是一款基于Canvas的HTML5图表应用,在图表数据初始化的时候伴随动画效果. 在线演示: 点击演示 源代码下载: 点击下载 核心jQuery代码: var myData = {   labels ...

  3. c++各类变量汇总

    一.局部变量和全局变量: (1)局部变量:局部变量也叫自动变量,它声明在函数开始,生存于栈,它的生命随着函数的返回而结束. #include <stdio.h> int main(void ...

  4. RoboGuice注入框架简单应用

    1.设置Activity为RoboActivity; 2.设置界面@ContentView(int resId) 3.使用@InjectView(int resId)反射组件 4.使用@Inject ...

  5. 获取Google音乐的具体信息(方便对Google音乐批量下载)

    Google音乐都是正版音乐, 不像百度所有都是盗链, 并且死链也多. 但有一个麻烦就是要下载Google音乐的时候得一个一个的点击下载链接, 进入下载页面再点"下载", 才干下载 ...

  6. C++设计模式之建造者模式(三)

    4.引入钩子方法的建造者模式 建造者模式除了逐步构建一个复杂产品对象外.还能够通过Director类来更加精细地控制产品的创建过程.比如添加一类称之为钩子方法(HookMethod)的特殊方法来控制是 ...

  7. hadoop在实现kmeans算法——一个mapreduce实施

    写mapreduce程序实现kmeans算法.我们的想法可能是 1. 次迭代后的质心 2. map里.计算每一个质心与样本之间的距离,得到与样本距离最短的质心,以这个质心作为key,样本作为value ...

  8. As long as Binbin loves Sangsang

    题目连接 题意: 给定一个无向图,每一个边有两个属性.长度和一个字母'L','O','V'.'E'中的一个.从1点開始到达n点,每次必须依照L -> O -> V -> E -> ...

  9. windows phone 7,sliverlight 下载网页的解析,关于wp7 gb2312编码

    原文:windows phone 7,sliverlight 下载网页的解析,关于wp7 gb2312编码 关于silverlight和wp7(windows phone 7)是默认不支持gb2312 ...

  10. hdu4585 &amp; BestCoder Round #1 项目管理(vector应用)

    主题链接:pid=4858">http://acm.hdu.edu.cn/showproblem.php?pid=4858 项目管理 Time Limit: 2000/1000 MS ...