js中闭包的概念
闭包并不是 JavaScript 特有的,大部分高级语言都具有这一能力。
什么是闭包?
A closure is the combination of a function bundled together (enclosed) with references to its surrounding state (the lexical environment).
这段是 MDN 上对闭包的定义,理解为:一个函数及其周围封闭词法环境中的引用构成闭包。可能这句话还是不好理解,看看示例:
function createAction() { var message = "封闭环境内的变量"; return function() { console.log(message); } } const showMessage = createAction(); showMessage(); // output: 封闭环境内的变量
这个示例是一个典型的闭包,有这么几点需要注意:
-
showMessage
是createAction
执行后从中返回出来的一个函数。 -
createAction
内部是一个封闭的词法环境,message
作为该封装环境内的变量,在外面是绝不可能直接访问。 -
showMessage
在createAction
外部执行,但执行时却访问到其内部定义的局部变量message
(成功输出)。这是因为showMessage
引用的函数(createAction
内部的匿名函数),在定义时,绑定了其所处词法环境(createAction
内部)中的引用(message
等)。 - 绑定了内部语法环境的匿名函数被
return
带到了createAction
封闭环境之外使用,这才能形成闭包。如果是在createAction
内部调用,不算是闭包。
好了,我相信 1, 2, 4 都好理解,但是要理解最重要的第 3 点可能有点困难 —— 困难之处在于,这不是程序员能决定的,而是由语言特性决定的。所以不要认为是“你”创建了闭包,因为闭包是语言特性,你只是利用了这一特性。
如果语言不支持闭包,类似上面的代码,在执行 showMessage
时,就会找不到 message
变量。我特别想去找一个例子,但是很不幸,我所知道的高级语言,只要能在函数/方法内定义函数的,似乎都支持闭包。
把局部定义的函数“带”出去
前面我们提到了可以通过 return
把局部定义的函数带出去,除此之外有没有别的办法?
函数在这里已经成为“货”,和其他货(变量)没有区别。只要有办法把变量带出去,那就有办法把函数带出去。比如,使用一个“容器”对象:
function encase(aCase) { const dog = "狗狗"; const cat = "猫猫"; aCase.show = function () { console.log(dog, cat); }; } const myCase = {}; encase(myCase); myCase.show(); // output: 猫猫 狗狗
是不是受到了启发,有没有联想到什么?
模块和闭包
对了,就是 exports 和 module.exports。在 CJS (CommonJS) 定义的模块中,就可以通过 exports.something
逐一带货,也可以通过 module.exports = ...
打包带货,但不管怎么样,exports
就是带货的那一个,只是它有可能是原来安排的 exports
也可能是被换成了自己人的 exports
。
ESM (ECMAScript Module) 中使用了 import
和 export
语法,也只不过是换种方法带货出去而已,和 return
带货差不多,区别只在于 return
只能带一个(除非打包),export
可以带一堆。
还要补充的是,不管是 CJS 还是 ESM,模块都是一个封装环境,其中定义的东西只要不带出去,外面是访问不到的。这和网页脚本默认的全局环境不同,要注意区别。
如果用代码来表示,大概是定义模块的时候以为是这样:
const var1 = "我是一个顶层变量吧"; function maybeATopFunction() { }
结果在运行环境中,它其实是这样的(注意:仅示意):
// module factory function createModule_18abk2(exports, module) { const var1 = "我是一个顶层变量吧"; function maybeATopFunction() { } } // ... 遥远的生产线上,有这样的示意代码 const module = { exports: {} }; const m18abk2 = createModule_18abk2(module) ?? module; // 想明白 createModule_18abk2 为什么会有一个随机后缀没?
还是那个函数吗?
扯远了,拉回来。思考一个问题:理论上来说,函数是一个静态代码块,那么多次调用外层函数返回出来的闭包函数,是同一个吗?
试试:
function create() { function closure() { } return closure; } const a = create(); const b = create(); console.log(a === b); // false
如果觉得意外,那把 closure()
换种方式定义看会不会好理解一点:
function create() { closure = function() { } return closure; }
如果还不能理解,再看这个:
function create() { const a = function () { }; const b = function () { }; console.log(a === b); // false }
能理解了不:每一次 function
都定义了一个新的函数。函数是新的,名字不重要 —— 你能叫小明,别人也能叫小明不是。
所以,总结一下:
闭包是由一个函数以及其定义时所在封闭环境内的各种资源(引用)构成,拿到的每一个闭包都是独一无二的,因为构成闭包的环境资源不同(不同的局部环境,定义了不同的局部变量,传入了不同的参数等)。
闭包,这回搞明白了!
推荐教程:《JS教程》
以上是js中闭包的概念的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

热门话题

在C++中,闭包是能够访问外部变量的lambda表达式。要创建闭包,请捕获lambda表达式中的外部变量。闭包提供可复用性、信息隐藏和延迟求值等优势。它们在事件处理程序等实际情况中很有用,其中即使外部变量被销毁,闭包仍然可以访问它们。

C++Lambda表达式支持闭包,即保存函数作用域变量并供函数访问。语法为[capture-list](parameters)->return-type{function-body}。capture-list定义要捕获的变量,可以使用[=]按值捕获所有局部变量,[&]按引用捕获所有局部变量,或[variable1,variable2,...]捕获特定变量。Lambda表达式只能访问捕获的变量,但无法修改原始值。

闭包是一种嵌套函数,它能访问外层函数作用域的变量,优点包括数据封装、状态保持和灵活性。缺点包括内存消耗、性能影响和调试复杂性。此外,闭包还可以创建匿名函数,并将其作为回调或参数传递给其他函数。

标题:闭包引起的内存泄漏及解决方法引言:闭包是JavaScript中一个非常常见的概念,它可以让内部函数访问外部函数的变量。然而,闭包在使用不当的情况下可能导致内存泄漏。本文将探讨闭包引起的内存泄漏问题,并提供解决方法及具体代码示例。一、闭包引起的内存泄漏问题闭包的特性是内部函数可以访问外部函数的变量,这意味着在闭包中引用的变量不会被垃圾回收。如果使用不当,

函数指针和闭包对Go性能的影响如下:函数指针:稍慢于直接调用,但可提高可读性和可复用性。闭包:通常更慢,但可封装数据和行为。实战案例:函数指针可优化排序算法,闭包可创建事件处理程序,但会带来性能损失。

是的,可以通过链式调用和闭包优化代码简洁性和可读性:链式调用可将函数调用链接为一个流畅接口。闭包可创建可重用代码块,并在函数外部访问变量。

Java中的闭包允许内部函数访问外部的作用域变量,即使外部函数已经退出。通过匿名内部类实现,内部类持有一个外部类的引用,使外部变量保持活动。闭包增强了代码灵活性,但需要注意内存泄漏风险,因为匿名内部类对外部变量的引用会保持这些变量的活动状态。

Go语言函数闭包在单元测试中发挥着至关重要的作用:捕获值:闭包可以访问外部作用域的变量,允许在嵌套函数中捕获和重用测试参数。简化测试代码:通过捕获值,闭包消除了对每个循环重复设置参数的需求,从而简化了测试代码。提高可读性:使用闭包可以组织测试逻辑,使测试代码更清晰、更易于阅读。
