Javascript闭包
维基百科:在计算机科学中,闭包(Closure),是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。
上面的解释难免有些抽象,为了化繁为简,本文将通过实例的方式,探究Javascript中闭包的概念及其用途。为了更好地理解闭包,我将从Javascript的变量的作用域谈起。
一、Javascript变量的作用域
有点类似于原型链(proto chain),Javascript中变量遵从作用域链(scope chain)规则。
如上图所示,在Javascript中,每一个函数体对应于一个作用域。 当访问一个变量时,我们会先访问当前作用域内是否有定义该变量,如果没有就会在该作用域外的作用域内寻找是否有改变量,依此类推,一直寻找到全局变量。如果全局变量中依旧没有定义该变量,就会返回undefined。
我们来看下下面这个例子:
var milk = '外面的特仑苏'
function wrapper1() {
var milk = '里面的特仑苏'
console.log('我要喝' + milk) //我要喝里面的特仑苏
}
function wrapper2() {
console.log('我要喝' + milk) //我要喝外面的特仑苏
}
wrapper1()
wrapper2()
在上述例子中,我们在wrapper1函数体内定义了变量milk,因此wrapper1在寻找完当前作用域即可以得到里面的特仑苏,而在wrapper2函数体内没有定义变量milk,它会沿着作用域链去寻找全局变量,然后得到了外面的特仑苏。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。
二、闭包的概念
按照作用域链的规则,我们无法在某函数体外访问到该函数内的局部变量。但是出于某些目的我们想在函数体外访问到函数体内的局部变量,我们该怎么做呢?请看下面例子:
function wrapper1() {
var milk = '里面的特仑苏'
function drink() {
console.log('我喝了' + milk)
}
return drink
}
var result = wrapper1()
result() //我喝了里面的特仑苏
我们在函数体内再创建一个函数,并且把这个内部函数drink作为外部函数wrapper1的返回值。我们通过执行函数wrapper1获得了它的返回值drink,并且执行它,就成功的访问到了它的内部变量milk(里面的特仑苏)。
回想维基百科中闭包的定义,再结合上述例子:drink函数就是一个闭包,因为它引用了处于它外部的变量milk。这个被引用的外部变量milk和函数drink一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。
三、闭包的用途
从上述例子中不难看出,闭包的一个用途就是可以访问函数的内部变量。从而可以实现一些面向对象的功能,例如设置类的隐私变量,关于这一点可以参考《对Javascript 类、原型链、继承的理解》。
闭包的另一个用途就是可以使变量一直保存在内存之中,不被垃圾回收机制所回收。看下面这个例子:
var change;
function wrapper() {
var milk = '特仑苏'
function drink() {
console.log('我喝了' + milk)
}
change = function () {
milk = 'AD钙奶'
}
return drink
}
var result = wrapper()
result() //我喝了特仑苏
change()
result() //我喝了AD钙奶
可以看到,wrapper执行之后,milk变量一直能被访问到,原因就是result引用了wrapper内部的drink函数,drink函数又引用了milk变量,因此它一直不会被垃圾回收机制所回收。
四、慎用闭包
因为闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则可能会造成内存泄露。解决方法是,在使用完闭包函数之后,将变量设置为undefined。比如在上例中,在使用完result之后,将result设置为null或者undefined。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
-
上一篇
6月20日云栖精选夜读丨国内首家!阿里云宣布全面提供IPv6服务
IPv6作为下一代互联网的技术基础,对物联网、车联网、人工智能等新兴产业的发展有着重大影响。6月20日,中国云计算的领头羊阿里云宣布联合三大运营商全面对外提供IPv6服务,希望能在2025年前帮助中国互联网真正实现“IPv6 Only”。 热点热议 国内首家!阿里云宣布全面提供IPv6服务 作者:阿里云头条 发表在:云栖学习小组 从本科生到数据科学家,为啥这个职业门槛高? 作者:异步社区 Rich Caruana:压缩深度神经网络模型让你兼得可解释性与准确性! 作者:技术小能手 发表在:数据派THU 知识整理 Java中的路径问题 作者:codingcoge JavaScript设计模式:单例 作者:小丑丁丁 基于Hadoop的数据分析平台搭建 作者:技术小能手 发表在:中生代技术 TensorFlow教程:快速入门深度学习五步法(附Keras实例) 作者:技术小能手 发表在:数据派THU Python 4 种不同的存取文件骚操作 作者:技术小能手 发表在:小詹学Python 美文回顾 TextRank算法抽取关键词 作者:citibank 从MySQL到HBase:数据存储方案转型的...
-
下一篇
Java集合框架源码解析之LinkedHashSet
阅读本节内容需要读者对 HashMap 、HashSet 和 LinkedHashMap 的源码有所了解,因为 LinkedHashSet 的内部实现都是来自于这三个容器类,其内部源码十分简单,简单到它只有一个成员变量、四个构造函数、一个 Set 接口的方法 如果你想多了解下这三个容器类,可以从这里获得:Java集合框架源码解析 LinkedHashSet 的所有源码如下所示 package java.util; public class LinkedHashSet<E> extends HashSet<E> implements Set<E>, Cloneable, java.io.Serializable { //序列化ID private static final long serialVersionUID = -2851667679971038690L; //自定义初始容量与装载因子 public LinkedHashSet(int initialCapacity, float loadFactor) { super(initialCapac...
相关文章
文章评论
共有0条评论来说两句吧...