当前位置: 首页 > 编程日记 > 正文

javascript中重要概念-闭包-深入理解

在上次的分享中javascript--函数参数与闭包--详解,对闭包的解释不够深入。本人经过一段时间的学习,对闭包的概念又有了新的理解。于是便把学习的过程整理成文章,一是为了加深自己闭包的理解,二是给读者提供学习的途径,避免走弯路。

在javascript--函数参数与闭包--详解这篇文章中,我详细介绍了闭包的概念。以下的分享对闭包的基本概念只会稍稍带过。如果对闭包的概念不熟悉的同学,请移步至javascript--函数参数与闭包--详解。

以下的分享会分为如下内容:

1.let命令

2.闭包特点的解读

3.循环中的闭包

1.let命令

  在讲闭包前,有必要谈谈ES6中的新概念,let命令。因为在赘述循环中的闭包时会使用到let命令。

基本用法

  ES6新增了let命令,用来声明变量。它的用法类似于var,但是所声明的变量,只在let命令所在的代码块内有效。

复制代码
1     if (true) {
2         var a = 1;
3         let b = 2;
4     }
5     console.log(a); // 1
6     console.log(b); // ReferenceError: b is not defined
复制代码

在javascript--函数参数与闭包--详解中,谈到在局部变量只能在函数内部声明,在其他代码(如 if 条件语句,for循环语句)用 var 声明的变量都为全局变量。

在上面代码中,分别用 let和 var 声明了两个变量。然后在代码块之外调用这两个变量,结果 let 声明的变量报错,var 声明的变量返回了正确的值。这表明,if 条件语句中使用var声明的变量为全局变量,可以在全局作用域下访问。而 let 声明的变量只在它所在的代码块有效,在全局作用域下无法访问。

再来看看这两个例子。

1     for (let i = 0; i < 10; i++) {}
2     console.log(i);  //ReferenceError: i is not defined
1     for (var i = 0; i < 10; i++) {}
2     console.log(i);  // 10

2.闭包特点的解读

  我们知道,闭包有三个特点

  a:在一个函数内部定义另外一个函数。

  b:内部函数可以访问外部函数定义的局部变量 (变量采用var声明)

  c:让局部变量始终保存在内存中。也就是说,闭包可以使得它诞生的环境一直存在。

  我们来看一个例子,尝试串起这三个特点。

复制代码
 1     function keith() {2         var a = 1;3         return function() {4             return a++;5         }6     }7     var result = keith();8     console.log(result()); //19     console.log(result()); //2
10     console.log(result()); //3
复制代码

首先,在函数keith内部返回了一个匿名函数,如果函数keith没有返回值,则默认返回值为undefined。

然后,因为在函数keith中返回了一个匿名函数,又把调用函数keith的结果赋值给了全局变量result,所以全局变量result是一个闭包。当连续调用result时,依次返回1,2,3。返回值说明了内部函数可以访问外部函数定义的局部变量。也就是说,闭包记住了外部函数定义的局部变量的调用结果。

最后,因为我们把一个闭包赋值给了一个全局变量result,在调用时依次输出1,2,3。说明了在函数keith外部访问的这个局部变量a一直存在全局作用域中。也就是说,局部变量 a 一直存在于内存当中,所以不会被垃圾回收机制回收。

所以使用闭包的时候的注意点:

由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。

3.循环中的闭包

  一个常见的错误出现在循环中使用闭包,假设我们需要在每次循环中调用循环序号。

1     for (var i = 0; i < 10; i++) {
2         setTimeout(function() {
3             console.log(i); //10
4         }, 1000)
5     }

上面代码中,不会符合我们的预期,输出数字0-9。而是会输出数字10十次。

出现错误的原因在于我们在setTimeout函数里面定义了一个匿名函数,匿名函数的作用是在控制台输出变量 i,而变量 i 是一个全局变量,在全局范围内都有效。所以每一次循环,新的 值都会覆盖旧值,导致最后输出的是最后一轮的i的值。

所以,针对循环中的闭包,有以下两种解决方法。

一是使用立即执行函数(IIFE),并把 i 作为它的参数,此时函数内 e 变量就拥有了 i 的一个拷贝。当传递给 setTimeout 的匿名函数执行时,它就拥有了对 e 的引用,而这个值是不会被循环改变的。

复制代码
1     for (var i = 0; i < 10; i++) {
2         (function(e){
3             setTimeout(function() {
4                 console.log(e); //1,2,3,....,10
5             }, 1000)
6         })(i)
7     }
复制代码

二是让变量 i 只在代码块中有效。也就是说让其成为局部变量。变量 是 let 声明的,当前的 只在本轮循环有效,所以每一次循环的 其实都是一个新的变量,所以最后输出的是1,2,3,4....,10。

1     for (let i = 0; i < 10; i++) {
2         setTimeout(function() {
3             console.log(i); //1,2,3...,10
4         }, 1000)
5     }

相关文章:

ssl握手过程和ca证书验证

转载&#xff1a;https://www.cnblogs.com/cposture/p/9029014.html SSL 认证 可以将 SSL 服务器与客户端之间的通信配置为使用单向或双向 SSL 认证。 单向 SSL 认证一般是客户端利用服务器传过来的信息验证服务器的合法性&#xff0c;服务器的合法性包括&#xff1a;证书是…

【ACM】练武奇才

题目链接&#xff1a;http://acm.nuc.edu.cn/OJ/contest/show/43/1005 【问题描述】 很久很久以前&#xff0c;constbh大神还在上着小学。一天&#xff0c;在放学的路上&#xff0c;他被一位乞丐叫住&#xff0c;这位乞丐对constbh说&#xff0c;我看你骨骼惊奇&#xff0c;…

Bat命令学习

参考资料&#xff1a;http://www.cnblogs.com/SunShineYPH/archive/2011/12/13/2285570.html

记一次CentOS7内核kernel的删除重装

人生在于折腾&#xff0c;学习Linux更要多多折腾。在一次折腾中吸取教训&#xff0c;更易于记忆。今天我们来折腾Linux的内核&#xff1a;删除系统内核后&#xff0c;通过光盘进行kernel的重安装。友情提示&#xff1a;请在虚拟机环境进行&#xff0c;折腾前务必做好系统快照。…

tcpdump抓包并保存到远程服务器

有的时候&#xff0c;运行tcpdump抓包进程的主机A可能没有足够的硬盘空间。例如我们使用树霉派搭建了一个热点&#xff0c;然后我们想在树霉派上抓包&#xff0c;因为树霉派的存储很小&#xff0c;所以很容易在短时间内将存储空间使用完。 为了解决该问题&#xff0c;我们可以…

【ACM】家喻户晓的中药店(待更)

题目链接&#xff1a;http://acm.nuc.edu.cn/OJ/contest/show/43/1007 【问题描述】 long_xiao和const_hhh是一对恩爱的夫妻。 他们在京城经营着一家中药店&#xff0c;夫妻二人医术精湛、古道热肠&#xff0c;虽然年过花甲&#xff0c;身体依然硬朗。更重要的是&#xff…

常用MySQL的命令集锦

常用MySQL的命令集锦 一、连接MySQL 格式&#xff1a; mysql -h主机地址 -u用户名 &#xff0d;p用户密码 1、例1&#xff1a;连接到本机上的MYSQL。 首先在打开DOS窗口&#xff0c;然后进入目录 mysqlbin&#xff0c;再键入命令mysql -uroot -p&#xff0c;回车后提示你输密码…

google的gn构建系统

什么是GN&#xff1f; GN是一个生成Ninja构建文件的元构建系统&#xff0c;以便你可以用Ninja构建Chromium。 你为什么从GYP切换&#xff1f; 我们相信GN文件比GYP文件更具可读性和可维护性。GN很快&#xff1a; GN比GYP快20倍。GN支持作为构建的一部分&#xff0c;根据Ninj…

【ACM】五子棋

题目链接&#xff1a;http://acm.nuc.edu.cn/OJ/contest/show/25/1009 【问题描述】 五子棋想必大家都玩过&#xff0c;如果没有那我只能重新介绍一下规则了&#xff0c;当横竖斜&#xff08;共八个方向&#xff09;出现5个同色棋子时&#xff0c;则认为该颜色棋的选手获胜。…

源码编译安装httpd及其常见错误

一、编译安装的整体步骤 1、在官网下载源码&#xff0c;并解压2、切换到其目录中 3、执行./configure4、编译 二、编译中及安装后配置常见的参数及其说明编译中配置1&#xff09;指定安装路径--prefix/usr/local/Pacakage_name 指定安装路径--sysconfigdir/etc/Package_name …

Nhibernate3循序渐进(三): 一对多映射和级联保存

我们知道, 对于数据库中的一对多关系, 我们在建表的时候, 应该在多表这里建立外键我们准备这样一个场景, 大学里的系和学生, 一个系有多个学生Department类和Xml如下:Department.csusing System;using System.Collections.Generic;using System.Linq;using System.Text;namespa…

自定义UISearchBar外观

本文转载至 http://www.jianshu.com/p/66b5b777f5dc 最近&#xff0c;在项目过程中遇到要自定义SearchBar的外观&#xff0c;虽然自己觉得用系统默认的外观就行了&#xff0c;不过UI设计师要求不用系统的默认样式&#xff0c;要跟app主题保持 一致。 图1&#xff1a;设计效果图…

linux内存管理和原理分析

https://blog.csdn.net/rebirthme/article/details/50402082

【数据结构】邻接矩阵及其实现

文件操作比直接输入方便很多 直接输入&#xff1a; //建立图的邻接矩阵储存结构 #include <stdio.h> #include <string.h> #define M 20 #define FINITY 5000 typedef struct {char vexs[M];int edge[M][M];int n,e; }Mgraph;//c0&#xff0c;表示建立无向图 …

对职业生涯的思考

从刚毕业到目前所在公司&#xff0c;差不多6年了&#xff0c;想想这六年里面&#xff0c;自己的能力和刚毕业比有了很大的提升&#xff0c;但是现在在什么能力上&#xff0c;我不知道&#xff0c;毕竟没有去过别的公司。最近也在思考自己未来&#xff0c;算是比较迷茫阶段。趁最…

jquery源码分析(七)——事件模块 event(二)

上一章节探讨了事件的一些概念&#xff0c;接下来看下jQuery的事件模块。 jQuery对事件的绑定分别有几个API&#xff1a;.bind()/.live()/.delegate()/.on()/click()&#xff0c; 不管是用什么方式绑定&#xff0c;归根到底还是用addEventListener/attachEvent&#xff08;IE&a…

google gn构建系统的介绍

GN语言和操作 GN语言和操作 内容介绍 使用内置的帮助设计理念语言 字符串清单条件语句循环函数调用作用域和执行Scoping and execution命名事物 文件和目录名称构建配置目标CONFIGS 公共配置模板其他特性 Imports路径处理模式执行脚本与Blaze的区别和相似之处 介绍 本页面描述…

【数据结构】邻接表的储存结构 建立图的邻接表算法

【数据结构】邻接矩阵及其实现 一个图的邻接矩阵的表示是唯一的&#xff0c;但其邻接表表示不唯一&#xff0c;这是因为在邻接表结构中&#xff0c;各便表结点的链接次序取决于建立邻接表时的算法以及输入的次序。 一般而言邻接矩阵适合存储稠密图&#xff0c;邻接表适合存储…

报错:该字符串未被识别为有效的DateTime

报错&#xff1a;该字符串未被识别为有效的DateTime □ 背景 前端的搜索条件中包含关于时间的字符串&#xff0c;由jquery ui的datepicker产生时间字符串。 服务端对时间做了一次转换&#xff1a;DateTime.Parse(Request["时间字段"].ToString())。 搜索的时候没有选…

Nagios监控笔记上

Nagios软件介绍及服务端安装部署实战1. Nagios服务端安装1.1 准备3台服务器或者虚拟机器管理IP地址角色备注192.168.1.80Nagios监控服务器192.168.1.81Lamp服务器被监控的客户端服务器192.168.1.82Lamp服务器被监控的客户端服务器1.2 解决perl编译问题&#xff1a;后面编译的软…

liunx查看python的site-packages路径

有时候我们在liunx上想修改查看python的包路径可以试试以下命令 from distutils.sysconfig import get_python_lib print(get_python_lib()) 如图&#xff1a;

【ACM】杭电OJ 2010

注意格式&#xff01;&#xff01;&#xff01;注意格式&#xff01;&#xff01;&#xff01; 空格的设置 \n的设置 #include <stdio.h> int main () {int i,m,n,g,s,b,flag;while(scanf("%d%d",&m,&n)!EOF){flag0;for(im;i<n;i){gi%10;bi/100…

中科院 工程硕士专业课 复试考试前的辅导安排

同学们大家好&#xff1a;学校定于12月6日、7日组织专业课辅导&#xff0c;1月初进行专业课复试及资格审查。辅导具体日程安排如下&#xff1a;12月6日下午13:00 数据结构&#xff08;报考软件工程、计算机技术领域考生&#xff09; 人文楼教一阶12月7日上午9:00 信号与系统…

TCP性能和发送接收Buffer的关系

本文希望解析清楚&#xff0c;当我们在代码中写下 socket.setSendBufferSize 和 sysctl 看到的rmem/wmem系统参数以及最终我们在TCP常常谈到的接收发送窗口的关系&#xff0c;以及他们怎样影响TCP传输的性能。 先明确一下&#xff1a;文章标题中所说的Buffer指的是sysctl中的 …

PHP 异常类 Exception 高洛峰 细说PHP

/** 1.自定义的异常类&#xff0c;必须是系统类Exception的子类* 如果继承Exception类&#xff0c;重写了构造方法,一定要调用一下父类的构造方法。*/class MyException extends Exception{//必须继承Exception类function __construct($mess){parent::__construct($mess);}func…

【ACM】杭电OJ 2023

注意最后又两个\n #include <iostream> #include <cstdio> #include <cstring> using namespace std; const int maxn 1000; int a[maxn][maxn]; double grade[maxn]; double average[maxn]; int main () {int m,n,i,j,flag,count;//n个学生&#xff0c;m门…

Hadoop学习笔记—7.计数器与自定义计数器

一、Hadoop中的计数器 计数器&#xff1a;计数器是用来记录job的执行进度和状态的。它的作用可以理解为日志。我们通常可以在程序的某个位置插入计数器&#xff0c;用来记录数据或者进度的变化情况&#xff0c;它比日志更便利进行分析。 例如&#xff0c;我们有一个文件&#x…

win10安装spacemacs

1、下载emacs最新版 26.1 2、解压emacs到你的安装目录,我的系统是D:/Program File/。执行/bin目录下的addpm.exe 这一步会在开始菜单创建快捷方式 3、在系统环境变量中添加新项HOME(具体环境变量设置方式请自行google)&#xff0c;该变量的路径决定了emacs启动时.emacs.d目录…

【ACM】杭电OJ 2024

注意&#xff1a; 1、getchar() 2、scanf和gets的区别 3、判断条件 C语言的合法标识符 1、由字母&#xff0c;数字&#xff0c;下划线组成 2、且首字符不能是数字 #include <iostream> #include <cstdio> #include <cstring> using namespace std; in…

王振的开发板_Android

任务一&#xff1a; 任务内容&#xff1a;主界面框架的搭建 发布时间&#xff1a;2016-8-26 已完成 任务二&#xff1a; 任务内容&#xff1a;我的 主界面的开发 发布时间&#xff1a;2016-8-27 已完成 任务三&#xff1a; 任务内容&#xff1a;发布界面 动画开发 发布时间&a…