关于JS中闭包的问题
一直以来,我都以为我已经懂了JavaScript中闭包的概念,直到有一次小伙伴突然问我这个概念的时候,我才发现我根本不知道该怎来么跟他来讲述这个概念。
那时候我就知道我是自我欺骗,打肿脸充胖子了。
所以,花了点时间去专门了解了一下,今天专门记录一下自己所理解的闭包。
一. 概念
闭包,简单来讲,就是定义在函数内部的函数,使用闭包,可以让你有权访问另一个函数作用域内的变量。
所以,想要了解闭包的前提是,你首先要知道在JS中变量作用域的问题。
创建闭包的常见方式就是在函数内部去创建另一个函数:
function fun() {
var variable = 'Hello World';
function inner() {
console.log(variable);
}
return inner;
}
var outer = fun();
outer(); // Hello World
在这个例子中,我们想在外部用到`fun()`中定义的`variable`的值,但是因为变量作用域的问题,我们不可能直接取到。
所以我们采取了变通的方法:在fun()函数内部又创建了一个函数inner(),这时fun()内部的variable对于inner()来说是可见的,既然inner()可以取到fun()中的变量,那么我们将inner()返回,就可以用到fun()中定义的variable了。
闭包在此处,就是链接函数内部和外部的一个桥梁。
在这里提一句:如果inner()内部存在新设置的变量,对于fun()函数来说是不可见的,此处涉及到JS中作用链的问题,理解作用链对于彻底理解闭包的问题很有帮助,可以参考JavaScript高级程序设计(第四章)去了解一下作用链。
其实闭包的定义也就这么简单,对于那些过于抽象的定义,置之不理即可,不用强迫自己去理解那些比较晦涩难懂的专业定义,记住自己最终的目的并不是为了咬文嚼字,实用才是根本。
最后借用知乎上一个回答来形象的描述一下闭包的概念:

二. 闭包的用处
我总结的闭包主要用处:
- 让外部可以读取函数内部的变量。
- 可以封装对象的私有属性和私有方法。
第一点用处就是在说闭包概念时候所举的例子。
下面说下第二点用处:可以封装对象的私有属性和私有方法。
function Worker(name) {
var _salary;
function setSalary(value) {
_salary = value;
}
function getSalary() {
return _salary;
}
return {
name: name,
setSalary: setSalary,
getSalary: getSalary
}
}
var cxk = Worker('cXK');
cxk.setSalary(100);
cxk.getSalary(); // 100
在上面的代码中,通过闭包,`_salary`变成了`cxk`的私有变量。
# 三. 需要注意的地方
第一点需要注意的地方是关于使用闭包时内存的问题,因为闭包会携带包含它的函数的作用域,因此会比其他的函数占用更多的内存,滥用闭包会造成网页的性能问题,所以对于闭包,建议只在绝对必要时在考虑使用。
对于闭包中垃圾回收的详细测试,参考js闭包测试/司徒正美。
第二点需要注意的就是在创建闭包时可能会常犯的错误:在循环中的闭包创建问题。
function createArray() {
var result = [];
for (var i = 0; i < 10; i++) {
result[i] = function () {
return i;
}
}
return result;
}
var arr = createArray();
arr[1](); // 10
arr[2](); // 10
可以看到,跟我们预期达到的结果不一样,每一个位置上的函数都返回了10。
这是因为每一个result[i]上都保存着createArray()函数的活动对象(参考JS中的作用链),而给result[i]进行赋值时,'function(){return i}'没有执行,所以最后在arr[1]运行时,返回的i其实都是同一个值,即最后生成的i,值为10。
可以做出如下修改。
修改一:在闭包里再添加一个闭包函数,同时立即执行。
function createArray() {
var result = [];
for (var i = 0; i < 10; i++) {
result[i] = function (num) {
return function () {
return num;
}
}(i)
}
return result;
}
var arr = createArray();
arr[1]();
修改二:修改`var`为`let`。
```javascript
function createArray() {
var result = [];
for (let i = 0; i var arr = createArray();
arr1;
<br>
以上就是我对闭包的比较浅显的认知,如果有不对的地方,希望能够指正,以免我误人子弟,谢谢。
关于JS中闭包的问题的更多相关文章
- 关于js中闭包的理解
1.以前很不理解js中闭包的概念及使用,下面来看一下 function foo() { var a = 123; var b = 456; return function () { return a; ...
- 彻底搞清js中闭包(Closure)的概念
js中闭包这个概念对于初学js的同学来说, 会比较陌生, 有些难以理解, 理解起来非常模糊. 今天就和大家一起来探讨一下这个玩意. 相信大家在看完后, 心中的迷惑会迎然而解. 闭包概念: 闭包就是有权 ...
- js中闭包和对象相关知识点
学习js时候,读到几篇不错的博客.http://www.cnblogs.com/yexiaochai/p/3802681.html一,作用域 和C.C++.Java 等常见语言不同,JavaScrip ...
- js中闭包的讲解
一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量 ...
- 对js中闭包,作用域,原型的理解
前几天,和朋友聊天,聊到一些js的基础的时候,有一种‘好像知道,好像又不不知道怎么讲的感觉’...于是捡起书,自己理一理,欢迎拍砖. 闭包 理解闭包首先要理解,js垃圾回收机制,也就是当一个函数被执行 ...
- JS中闭包的介绍
闭包的概念 闭包就是能够读取其他函数内部变量的函数. 一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascri ...
- [学习笔记]JS中闭包的理解
一.闭包概念的理解 闭包,又称为词法闭包或函数闭包指引用了自由变量的函数.这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外. 自由变量:该变量既不是函数本身定义的也不是函数 ...
- JS中闭包、函数与对象的介绍和用法
闭包 闭包概念:当一个内部函数被调用,就会形成闭包,闭包就是能够读取其他函数内部变量的函数,定义在一个函数内部的函,创建一个闭包环境,让返回的这个子程序抓住i,以便在后续执行时可以保持对这个i的引用. ...
- js中闭包来实现bind函数的一段代码的分析
今天研究了一下bind函数,发现apply和call还可以有这样的妙用,顺便巩固复习了闭包. var first_object = { num: 42 }; var second_object = { ...
随机推荐
- bhp 阅读笔记 OSX 下 setuptools pip 安装
安装 python-setuptools python-pip 尝试 brew install python-setuptools 失败 brew update 失败 $ cd `brew --pre ...
- webpack快速入门——集中拷贝静态资源
工作中会有一些已经存在但在项目中没有引用的图片资源或者其他静态资源(比如设计图.开发文档), 这些静态资源有可能是文档,也有可能是一些额外的图片.项目组长会要求你打包时保留这些静态资源, 直接打包到制 ...
- (转)C# Enum,Int,String的互相转换 枚举转换--非常实用
Enum为枚举提供基类,其基础类型可以是除 Char 外的任何整型.如果没有显式声明基础类型,则使用 Int32.编程语言通常提供语法来声明由一组已命名的常数和它们的值组成的枚举. 注意:枚举类型的基 ...
- 【HDU5126】 stars k-d树
题目大意:有$m$个操作,分两种:在指定三维坐标内加入一个点,询问指定空间内点的数量. 其中$m≤5*10^{4},1≤x,y,z≤10^9$ 这题几乎就是裸的$k-d$树啊.我们动态维护一棵$k-d ...
- celery问题记录
1. 问题:WARNING/MainProcess] /home/jihonghe/.virtualenvs/py3_dj217_env/lib/python3.6/site-packages/bil ...
- 漫谈NIO(3)之Netty实现
1.前言 上一章结合Java的NIO例子,讲解了多路IO复用的一个基本使用方法,通过实际编码加深对其理解.本章开始进入Netty的环节,前面两章都是为了Netty进行铺垫说明.此节将对比Java的NI ...
- Servlet Filter(过滤器)、Filter是如何实现拦截的、Filter开发入门
Servlet Filter(过滤器).Filter是如何实现拦截的.Filter开发入门 Filter简介 Filter也称之为过滤器,它是Servlet技术中最激动人心的技术,WEB开发人员通过F ...
- 【数组】Container With Most Water
题目: Given n non-negative integers a1, a2, ..., an, where each represents a point at coordinate (i, a ...
- CentOS6.5下Ambari安装搭建部署大数据集群(图文分五大步详解)(博主强烈推荐)
第一步: Ambari安装之Ambari安装前准备(CentOS6.5)(一) 第二步: Ambari安装之部署本地库(镜像服务器)(二) 第三步: Ambari安装之安装并配置Ambari-serv ...
- 解决linux安装软件依赖的曲线救国方案
相信大家在一台无法连接外网的linux上安装软件时,对于软件依赖的安装,都会特别头疼,因为软件依赖的安装,不论是其数量,还是安装的复杂度都比软件本身要高出一个维度! 今天就和大家分享一个,解决linu ...