闭包是指那些能够访问独立(自由)变量的函数 (变量在本地使用,但定义在一个封闭的作用域中)。换句话说,这些函数可以“记忆”它被创建时候的环境。——这句话其实有点难以理解。我觉得应该用一些例子来理解闭包的含义。

  • 闭包#1

先来看一个函数:

1
2
3
4
5
6
function f(){
var b = "b";
return function(){
return b;
}
}

这个函数含有一个局部变量b,它在全局空间里面是不可见的。而f()的返回值是另一个匿名函数,该函数有自己的私有的空间,可以访问f()的空间和全局空间,所以b对它来说是可见的。

1
2
var n = f();
n();//此时控制台输出"b"

因为f()是一个全局函数,它可以在全局空间被调用,当然也可以将其返回值赋值给另一个全局变量n,从而生成一个可以访问f()私有空间的新的全局函数。

  • 闭包#2

再来看看下面这个跟上面差不多的函数,在这里f()不再返回函数了,而是直接在函数体内创建一个新的全局函数

1
2
3
4
5
6
7
8
9
var n;
function f(){
var b = "b";
n = function(){
return b;
}
}
f();
n();//调用f(),然后再调用n(),控制台输出"b"

由于n()没有使用var语句,因此它是全局的,同时它也可以访问f()的作用域,所以哪怕n()最后升级为全局函数,但它依然可以保留对f()作用域的访问权。

  • 闭包#3

函数通常会让自身的参数视为局部变量,所以我们创建返回函数时,也可以令其返回父级函数的参数,例如:

1
2
3
4
5
6
7
8
9
function f(arg){
var n = function(){
return arg;
};
arg++;
return n;
}
var m = f(123);
m();//调用函数,输出124

有一点很重要:n被赋值时函数并没有被调用,调用发生是在n被求值,也就是执行return n;语句时。由此可以看出,函数所绑定的是作用域本身,而不是该作用域中的变量或变量当前所返回的值。

  • 循环中的闭包#4

这种闭包所导致的bug往往很难被发现,因为它们表面上看起来很正常,来看一下下面的函数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
function f(){
var a = [];
var i;
for (i = 0; i < 3; i++){
a[i] = function(){
return i;
}
}
return a;
}
//下面来运行一下函数,并将结果赋值给数组a
var a = f();
a[0]();//输出3
a[1]();//输出3
a[2]();//输出3

为啥不是0、1、2呢?为啥会这样呢?原来在这里创建的三个闭包,它们都指向了一个共同的局部变量i,但是,闭包不会记录它们的值,它们所拥有的的只是一个i的连接(即引用),因此只能返回i当前值,因为i结束循环时值为3,所以这三个函数都指向一个共同值3

如何纠正?显然,需要a[i]指向三个不同的变量,下面是解决方案之一:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
function f(){
var a = [];
var i;
for (i = 0; i < 3; i++){
a[i] = (function(x){
return function(){
return x;
}
})(i);
}
return a;
}
//下面来运行一下函数,并将结果赋值给数组a
var a = f();
a[0]();//输出0
a[1]();//输出1
a[2]();//输出2

这里使用了自调函数,不再直接返回i的值,而是将i传递给自调函数,i赋值给了局部变量x,这样一来,每次迭代x就会拥有各自不同的值了。

解决方案二:

1
2
3
4
5
6
7
8
9
10
11
12
13
function f(){
function aa(x){
return function(){
return x;
}
}
var a = [];
var i;
for (i = 0; i < 3; i++){
a[i] = aa(i);
}
return a;
}

方案二不使用自调函数,而是定义了一个内部函数实现相同的功能,每次迭代操作中,将i的值“本地化”。

  • Getter与Setter、 迭代器 闭包的应用示例#5
1
2
3
4
5
6
7
8
9
10
11
12
13
var getValue, setValue;
(function(){
var temp = 0;
getValue = function(){
return temp;
};
setValue = function(v){
temp = v;
}
})();
getValue();//输出0
setValue(123)
getValue();//输出123

这个例子是通过一个匿名自调函数来实现的,定义的全局函数setValue()和getValue(),确保局部变量temp的不可直接访问性。

下面是一个接受数组输入的初始化函数,其中定义了一个私有指针,该指针会指向数组中的下一个元素。

1
2
3
4
5
6
7
8
9
10
11
12
function setup(x){
var i = 0;
return function() {
return x[i++];
};
}
//调用setup(),创建我们所需的next()函数
var next = setup(['a', 'b', 'c']);
//重复调用next(),就可以不停的获取下一个元素
next();//输出"a"
next();//输出"b"
next();//输出"c"

到这里闭包就暂时告一段落了,以后有新的理解再写。

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

  1. 我从来不理解JavaScript闭包,直到有人这样向我解释它...

    摘要: 理解JS闭包. 原文:我从来不理解JavaScript闭包,直到有人这样向我解释它... 作者:前端小智 Fundebug经授权转载,版权归原作者所有. 正如标题所述,JavaScript闭包 ...

  2. 【转】深入理解JavaScript闭包闭包(closure) (closure)

    一.什么是闭包?"官方"的解释是:闭包是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.相信很少有人能直接看懂这句话,因为他描述 ...

  3. 深入理解JavaScript闭包【译】

    在<高级程序设计>中,对于闭包一直没有很好的解释,在stackoverflow上翻出了一篇很老的<JavaScript closure for dummies>(2016)~ ...

  4. 全面理解Javascript闭包和闭包的几种写法及用途

    好久没有写博客了,过了一个十一长假都变懒了,今天总算是恢复状态了.好了,进入正题,今天来说一说javascript里面的闭包吧!本篇博客主要讲一些实用的东西,主要将闭包的写法.用法和用途.  一.什么 ...

  5. 深入理解JavaScript闭包(closure)

    最近在网上查阅了不少javascript闭包(closure)相关的资料,写的大多是非常的学术和专业.对于初学者来说别说理解闭包了,就连文字叙述都很难看懂.撰写此文的目的就是用最通俗的文字揭开Java ...

  6. 深入理解javascript闭包(一)

    闭包(closure)是Javascript语言的一个难点.也是它的特色,非常多高级应用都要依靠闭包实现. 一.什么是闭包? 官方"的解释是:闭包是一个拥有很多变量和绑定了这些变量的环境的表 ...

  7. 深入理解javascript闭包(一)

    原文转自脚本之家(http://www.jb51.net/article/24101.htm) 闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. ...

  8. 深入理解Javascript闭包概念

    一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部能够直接读取全局变量 ...

  9. 轻松理解JavaScript闭包

    摘要 闭包机制是JavaScript的重点和难点,本文希望能帮助大家轻松的学习闭包 一.什么是闭包? 闭包就是可以访问另一个函数作用域中变量的函数. 下面列举出常见的闭包实现方式,以例子讲解闭包概念 ...

  10. 【译】理解JavaScript闭包——新手指南

    闭包是JavaScript中一个基本的概念,每个JavaScript开发者都应该知道和理解的.然而,很多新手JavaScript开发者对这个概念还是很困惑的. 正确理解闭包可以帮助你写出更好.更高效. ...

随机推荐

  1. poj3617 best cow line(贪心题)

    Best Cow Line Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 32687   Accepted: 8660 De ...

  2. Latex:插入伪代码

    *本文属于转载. *转载链接:https://blog.csdn.net/lwb102063/article/details/53046265 目录 clrscode algorithm algori ...

  3. MVC如何在解决方案下创建文件夹

    背景:为什么要在解决方案下创建文件夹? 比如,在开发过程中,会抽象出大量的公共方法,如数据库访问的方法.配置文件读取方法等等,将这些方法生成自己的DLL库文件,方便在其他的项目中进行复用.那么,这些方 ...

  4. golang连接orcale

    使用glang有一段时间了,最开始其实并不太喜欢他的语法,但是后来熟悉之后发现用起来还挺爽的.之前数据库一直使用mysql,连接起来没有什么问题,github上有很多完善的驱动,所以以为连接其他数据库 ...

  5. LA 3790 Overlapping Squares DFS

    题意: 给出一个字符矩阵,问能否是不超过6个2×2的正方形组成的. 分析: 每次找一个最表面的正方形然后DFS就好了. 一个正方形被移开后,用一个特殊符号标记上,下次再匹配的时候就直接忽略这些位置. ...

  6. bzoj3039 joyoi1939 玉蟾宫 悬线法

    悬线法 #include <iostream> #include <cstring> #include <cstdio> using namespace std; ...

  7. sqlserver常用知识点备忘录(持续更新)

    背景 一个项目的开发,离不开数据库的相关操作,表/视图设计,存储过程,触发器等等数据库对象的操作是非常频繁的.有时候,我们会查找系统中类似的代码,然后复制/粘贴进行再进行相应的修改.本文的目的在于归纳 ...

  8. 通用的高度可扩展的Excel导入实现(附Demo)

    Demo源码 背景 通过程序将excel导入到数据库中是一项非常常见的功能.通常的做法是:先将excel转成DataTable,然后将DataTable转换成List<T>,最终通过Lis ...

  9. Python 爬取图书图片和地址

    #-*- coding:utf-8 -*- import xlwt import urllib import re def getHtml(url): page = urllib.urlopen(ur ...

  10. Leetcode 477.汉明距离总和

    汉明距离总和 两个整数的 汉明距离 指的是这两个数字的二进制数对应位不同的数量. 计算一个数组中,任意两个数之间汉明距离的总和. 示例: 输入: 4, 14, 2 输出: 6 解释: 在二进制表示中, ...