javascript 近乎神话般的概念:闭包
写在前面
JavaScript 一个近乎神话 对于JavaScript有使用经验但却从未真正理解闭包概念的人来说,理解闭包可以说是某种意义上的重生。闭包并不是需要学习新的语法才能使用的工具。闭包的产生是基于词法作用域写代码时自然产生的结果。换句话说,你不需要要为了闭包而写闭包,闭包在我们写的代码中随处可见。当你真正了解闭包之后,会发现,哦~,原来我以前所敲的代码中已经出现了很多闭包了鸭!一个小 demo仔细看看下面的例子我们会感到奇怪,明明都是调用result(),为什么结果会不一样呢?
let count=500 //全局作用域
function foo1() {let count = 0;//函数全局作用域function foo2() {count++;//函数内部作用域console.log(count);return count;}return foo2;//返回函数
}
let result = foo1();
result();//结果为1
result();//结果为2
首先foo1()返回的是一个foo2()函数,当我们调用result()的时候就会返回foo2()执行的函数,foo2()里面有什么呢? 首先我们看到如下有一个count变量,但是没有定义.我们根据JavaScript的作用域链的定义可知,当函数内部的变量没有定义的时候,就会采用冒泡的方式,向上一级寻找.上一级没有接着上一级找,直到最顶层window. 如果都没有,就会报underfined的错误.这里我们在foo2()中找到了count,于是count+1,第一次输出的是1,没有什么问题.
function foo2() {count++;console.log(count);return count;}
但是第二次我们再执行result()的时候就出现了问题,为什么会是2呢?按照流程,首先再foo2()函数内部寻找count,没有然后到外层寻找,找到了count=0,这时候count+1应该为1才对.这里就涉及到闭包的问题了.
我目前是在职前端开发,如果你现在也想学习前端开发技术,
在入门学习前端的过程当中有遇见任何关于学习方法,学习路线,学习效率等方面的问题,
你都可以申请加入我的前端学习交流裙:前面:851 中间:231 最后:348。
里面聚集了一些正在自学前端的初学者裙文件里面也有我做前端技术这段时间整理的一些前端学习手册,
前端面试题,前端开发工具,PDF文档书籍教程,需要的话都可以自行来获取下载。
首先我们在原来的代码中加一个debugger,然后到谷歌浏览器右键检查,点击resources就可以看到右边有一个Closure,浏览器的可视化已经证实了这的确是一个闭包.并且count=1已经存储在了Closure之中.也就说明count=1没有被销毁,等下次在调用result()的时候count=2.
认识作用域
要学习Clusure必须了解JavaScript的作用域相关知识 作用域包括:1.全局作用域2.函数作用域4.块级作用域(es6 新出,解决 var 问题, 新增 let, const)
var count = 100; //全局作用域function foo1() {var count = 0; //函数全局作用域return count; //返回函数}if (count == 1) {//块级作用域console.log(count);}
上面代码简单可以看出作用域分类,需要注意是,一个函数(function)也是块级作用域,简单来说,一般有 {}都可以算做是一个块级作用域.
作用域链
作用域里面嵌套作用域,就形成了作用域链. 外部作用域无法访问内部的作用域,看如下例子
function foo(){
var n=1
function foo2(){var m=1console.log(n) //1
}
foo2()
}
foo()
console.log(n) //err: n is not defined
上述代码中在全局中无法访问内部的n,但是在嵌套的内部foo2()可以访问外部的函数,这就是作用域产生的特殊效果.
明白了作用域链,我们再来看个例子(很有迷惑性,仔细看看哦):
var name = 'Mike'; //第一次定义namefunction showName() {console.log(name); //输出 Mike 还是 Jay ?}function changeName() {var name = 'Jay'; //重新定义nameshowName(); //调用showName()}changeName();
上面的例子你觉得输出的是什么呢?答案是Mike.在这里我们引出了一个新的概念,词法作用域作用域有两种模型:
词法作用域(静态):js查找是按照代码书写时候的位置来决定的,而不是按照调用时候位置
动态作用域:目前还有使用的有Perl,Bash (可以自行了解)
通过词法作用域的的规则我们可以再来分析一下
调用changeName()时,找到这个函数
定义var name = “Jay”
调用showName()
在changeName()里面查找是否有showName()这个方法,发现没有,向外层查找,找到了
调用console.log(name),在函数内部查找有没有name,没有,向外查找,找到了,name=“Mike”
输出Mike
闭包
了解了上面的知识之后,终于来到了闭包
闭包在两本书上的官方解释:
1.小"黄"书(你不知道的JavaScript): 当函数可以记住并访问所在的词法作用域时,就产生了闭包,即使函数是在当前词法作用域之外执行.
2.红宝书(JavaScript高级程序设计): 闭包是指有权访问另一个 函数作用域中的变量的函数
非常抽象的一个概念,我自己的一个理解是:
当一个变量(就像上面的name)既不是该函数内部的局部变量,也不是该函数的参数,相对于作用域来说,就是一个自由变量(引用了外部变量),这样就会形成一个闭包.
怎么说呢?我们再来看看一开始我们使用的demo
let count = 500; //全局作用域
function foo1() {let count = 0; //函数全局作用域function foo2() {let count2 = 1; //随便新增一个变量// count++; 注释debugger;//console.log(count); 注释//return count; 注释}return foo2; //返回函数
}
let result = foo1();
result(); //结果为1
result(); //结果为2
再次使用浏览器看看,这时我们就发现Closure已经消失了,这也就证实我说的,如果函数内部不调用外部的变量,就不会形成闭包.但是如果调用了外部变量,那么就会形成闭包. 这也就是说不是所有的函数嵌套函数都能形成闭包
最后我们再来看一个循环闭包的例子
for (var i = 1; i <= 5; i++) {setTimeout(function timer() {debugger;console.log(i); // 输出什么?}, 1000);
}
答案 6 6 6 6 6 .因为setTimeout里面的回调函数是一个异步的过程(异步代表可以不用等待我这个代码先执行完,可以先往后执行),而for循环是同步的(代码只能从上往下的执行),立即执行,异步的setTimeout必须等待一秒才能执行,这时i早已经循环结束了.解决办法有三个:
1、将for循环中的var 改成let
for (let i = 1; i <= 5; i++) {setTimeout(function timer() {debugger;console.log(i); // 1 2 3 4 5}, 1000);
}
这样就没有问题了, 因为let是有块级的功能,每一层循环都是独立的,互不影响,所以才能正常输出.
2.把setTimeout()套上一个function
for (var i = 1; i <= 5; i++) {log(i); // 1 2 3 4 5
}
function log(i) {setTimeout(function timer() {debugger;console.log(i);}, 1000);
}
这样同样能够实现这个功能,原理和第一个方法一样,每一个log()都是独立的,互不影响,这样才能有正确的结果,var就是因为没有块级的功能,才会出问题 3. 包装成匿名函数
for (var i = 1; i <= 5; i++) {(function (i) {setTimeout(function timer() {debugger;console.log(i);}, 1000);})(i)
}
前面一个(func…)定义函数,后面一个(i)调用,这再JavaScript叫做立即执行函数,其实与第二种方式是一样的,只是写法不一样.
结语
理解JavaScript是一项重要的技能,在面试中也常常会有,这是迈进高级JavaScript工程师的必经之路.
javascript 近乎神话般的概念:闭包相关推荐
- javascript中关于作用域和闭包
列表项目 前言 学习了javascript已经很久了,关于这个语言中的这两个特性也是早已耳熟能详,但是在实际的使用的过程中或者是遇到相关的问题的时候,还是不能很好的解决. 因此我觉得很有必要深入的学习 ...
- JavaScript 函数(作用域以及闭包)
JavaScript 函数(作用域以及闭包) ・执行环境及作用域 执行环境定义了变量或函数有权访问的其他数据. 每个执行环境都有一个与之关联的变量对象,环境中定义的所有变量或函数都保存在这个对象中, ...
- 征服 JavaScript 面试:什么是闭包?| Eric Elliott
"征服 JavaScript 面试"是我写的一系列文章,来帮助面试者准备他们在面试 JavaScript 中.高级职位中将可能会遇到的一些问题.这些问题我自己在面试中也经常会问. ...
- 【你不知道的JavaScript上卷】——作用域与闭包
原文: [你不知道的JavaScript上卷]--作用域与闭包 JS语言万变不离其宗,其中最常用.最重要的也就是常用的几个大概念.数据类型.作用域.原型链.闭包.this指针.异步,不同的人理解不一样 ...
- JavaScript入门几个概念
JavaScript入门几个概念 刚刚入门JavaScript的时候,搞懂DOM.BOM以及它们的对象document和window很有必要. DOM是为了操作文档出现的API,document是它的 ...
- javascript(面向对象,作用域,闭包,设计模式等)
javascript(面向对象,作用域,闭包,设计模式等) 1. 常用js类定义的方法有哪些? 参考答案:主要有构造函数原型和对象创建两种方法.原型法是通用老方法,对象创建是ES5推荐使用的方法.目前 ...
- 浅谈Javascript -- 【嵌套函数及闭包】
这篇文章其实是要讲闭包的一些初级应用,但是为了将闭包,我们还是从嵌套函数开始说吧,纵使所有的JavaScript函数都可以说是闭包,但是只有当一个嵌套函数被导出到它所定义的作用域之外时,这种闭包才是有 ...
- JavaScript函数,作用域以及闭包
JavaScript函数,作用域以及闭包 1. 函数 (1). 函数定义:函数使用function关键字定义,它可以用在函数定义表达式或者函数声明定义. a. 函数的两种定义方式: * functio ...
- 传奇世界手游活人最多服务器,传奇世界手游:堪比神话般的万人攻城战,罪恶苍穹行会力敌全服!...
原标题:传奇世界手游:堪比神话般的万人攻城战,罪恶苍穹行会力敌全服! 大家好,今天小编呢为大家简单说一些手游的攻略.闲话少讲,咱们直接开始.大家玩游戏的都知道,游戏等级是非常要的,有些游戏到了一定的等 ...
最新文章
- 格式化_icecream_python
- 关于Linux路由表的route命令(转)
- 利用Bandwidth Splitter限制带宽
- rtc关机闹钟6 AlarmManagerService研究
- 2015年浪潮面试题
- 接口 java 1614953826
- 2017.9.21 道路修建(noi2011) 思考记录
- java整蛊小游戏源码_Java 开发打飞机小游戏(附完整源码)
- java键盘监听keyadapter_JAVA 键盘监听当中,释放键盘并没有停止运动
- squid不能启动问题及解决方法
- 金蝶ERP实现产品入库及委外加工冲减生产现场虚仓毛坯数(修正版07-05-10)
- Spring Boot 中使用WebJars引入javasript依赖
- asp.net高校宿舍后勤管理系统案例
- java field的方法_Java Field 详解
- 搜索 阿虚同学_凉宫春日阿虚台词“在虚构的故事当中寻求真实感的人脑袋一定有问题”动画是出自那一集?...
- 计算机基础知识统考考ppt么,(计算机应用基础统考)第一章计算机基础知识.ppt
- 服务器上传文件出现500错误,但是其他不涉及文件的接口均正常
- C语言 =(按位与后赋值)^=(按位异或后赋值) |=(按位或后赋值)
- scipy.sparse的一些整理
- 2016年 河南工业大学校赛 C题.魔法宝石