Javascript闭包

简介: 维基百科:在计算机科学中,闭包(Closure),是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。

维基百科:在计算机科学中,闭包(Closure),是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。

上面的解释难免有些抽象,为了化繁为简,本文将通过实例的方式,探究Javascript中闭包的概念及其用途。为了更好地理解闭包,我将从Javascript的变量的作用域谈起。

一、Javascript变量的作用域

有点类似于原型链(proto chain),Javascript中变量遵从作用域链(scope chain)规则。

scope_chain.png

如上图所示,在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。

参考资料:
1.闭包【维基百科】
2.学习使用Javascript闭包【阮一峰】

目录
相关文章
|
12天前
|
JavaScript
闭包(js的问题)
闭包(js的问题)
|
25天前
|
设计模式 JavaScript 前端开发
js开发:请解释闭包(closure)是什么,以及它的用途。
闭包是JavaScript中的关键特性,允许函数访问并操作外部作用域的变量,常用于实现私有变量、模块化和高阶函数。私有变量示例展示了如何创建无法外部访问的计数器;模块化示例演示了封装私有变量和函数,防止全局污染;高阶函数示例则说明了如何使用闭包创建能接收或返回函数的函数。
14 2
|
1月前
|
存储 缓存 JavaScript
|
1月前
|
自然语言处理 JavaScript 前端开发
探索JavaScript中的闭包:理解其原理与实际应用
探索JavaScript中的闭包:理解其原理与实际应用
17 0
|
1月前
|
自然语言处理 JavaScript 前端开发
深入理解JS的执行上下文、词法作用域和闭包(中)
深入理解JS的执行上下文、词法作用域和闭包(中)
|
1月前
|
存储 自然语言处理 JavaScript
深入理解JS的执行上下文、词法作用域和闭包(上)
深入理解JS的执行上下文、词法作用域和闭包(上)
|
3月前
|
前端开发 JavaScript 开发者
Web前端开发中的JavaScript闭包应用
JavaScript闭包是Web前端开发中常见的概念,它可以帮助开发者解决作用域问题,提高代码的可读性和可维护性。本文将介绍JavaScript闭包的基本概念和应用,以及如何在Web前端开发中使用闭包。
27 3
|
1月前
|
自然语言处理 前端开发 JavaScript
深入理解JavaScript中的闭包与作用域链
在JavaScript编程中,闭包和作用域链是两个非常重要的概念,它们对于理解代码的执行过程和解决一些特定问题至关重要。本文将深入探讨JavaScript中闭包和作用域链的原理和应用,帮助读者更好地理解这些概念并能够在实际项目中灵活运用。
|
1月前
|
JavaScript 前端开发
javascript闭包的理解(菜菜必看系列!!!)
javascript闭包的理解(菜菜必看系列!!!)
|
1月前
|
自然语言处理 JavaScript 前端开发
深入理解JS的执行上下文、词法作用域和闭包(下)
深入理解JS的执行上下文、词法作用域和闭包(下)