news 2026/6/10 11:47:19

闭包里的变量到底存哪了?图解 JS 词法环境与内存引用链路

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
闭包里的变量到底存哪了?图解 JS 词法环境与内存引用链路

拒绝玄学,看透本质:图解 JavaScript 词法环境与内存引用

很多同学在学习闭包时,往往只记住了“子函数可以使用父函数的变量”这个结论,却对底层的**“词法环境(Lexical Environment)”**知之甚少。

本文将剥离所有晦涩的术语堆砌,通过一段经典代码,深入内存底层,向你展示**代码逻辑(Code)、变量数据(Data)和执行环境(Environment)**在运行时的真实物理关系。

案例代码

我们以这段经典的闭包代码为例:

JavaScript

// 全局环境 let a = 100 function fun() { let timer = 0 // 局部变量 // 定义内部函数 function test() { timer++ console.log(timer) } return test } // 核心时刻:函数执行,并返回内部函数 const myTest = fun()

第一部分:直觉的误区(为什么你会困惑?)

初学者最容易犯的错误,是认为内存结构等同于代码结构。

错误的静态视角:

“因为代码里 test 嵌套在 fun 里,所以内存里 test 的数据也死死地包在 fun 里。fun 执行完,这块空间就该销毁,或者一直死板地存在那里。”

这种“俄罗斯套娃”式的直觉(如下图),无法解释为什么fun执行完了,timer还能被访问,也无法解释什么是“动态的作用域链”。

Code snippet

全局内存
let a = 100
fun 的内部空间 (静态嵌套)
let timer = 0
function test 代码

第二部分:揭秘真相——什么是“词法环境”?

要理解闭包,必须先理解 JavaScript 引擎执行代码时的分离存储原则。

在 JavaScript 运行时,“代码逻辑”和“状态数据”是分开存放的,靠“引用(指针)”连接。而管理这些状态数据的核心结构,就叫词法环境(Lexical Environment)

1. 核心概念定义

一个标准的词法环境由两部分组成:

  1. 环境记录器(Environment Record):这就是一个“登记表”,专门用来存放变量和函数的声明。比如timer = 0就记在这里。

  2. 外部环境引用(Outer Reference):这是一个“指针”,指向父级的词法环境。这就是作用域链的物理实体。

2. 运行时拆解:变量、函数、代码究竟在哪?

当执行const myTest = fun()这行代码时,内存中发生了极其精密的动态构建过程。

请配合下方的高维内存模型图来阅读:

  • 变量存在哪?(数据)

    当 fun() 被调用时,引擎在内存中创建了一个全新的词法环境(Lexical Environment)。变量 timer 的值 0 被保存在这个环境的环境记录器中。

  • 代码存在哪?(逻辑)

    函数 test 的代码逻辑(即函数体内的字符串)并不存储在词法环境里,而是存储在堆内存(Heap) 中一个独立的函数对象(Function Object)里。

  • 它们如何相互引用?(关键!)

    这是最关键的一步。当 test 函数对象被创建时,引擎会给它装上一个不可见的内部属性 [[Environment]]。这个属性直接指向了 fun 执行时创建的那个词法环境。


第三部分:一张图看懂内存物理结构

下面的 Mermaid 图解展示了fun()执行瞬间的真实内存快照。请重点关注代码存储(Heap)与环境记录(Stack/Context)的分离与连接。

Code snippet

堆内存 (存储函数实体/代码)
执行环境栈 (存储变量状态)
全局词法环境 (Global LE)
fun 函数词法环境 (Fun LE)
myTest 引用
闭包核心:
[[Environment]] 指针强引用
ƒ fun() 对象实体
Code: ...
ƒ test() 对象实体
Code: timer++ ...
-----------------------
[[Environment]]内部插槽
环境记录器:
timer: 0
Outer: -> Global LE
环境记录:
a: 100
fun: pointer -> func_fun
myTest: pointer -> func_test
Outer: null

第四部分:总结与顿悟

通过上图,我们可以回答最初的困惑:

  1. 为什么 fun 执行完,timer 不消失?

    看图中的红色粗线。虽然 fun 函数执行结束了,但返回的 test 函数对象里有一个 [[Environment]] 指针,死死地拉住了 fun 的词法环境。只要 myTest 还在引用 test 对象,这个Fun LE方块就永远无法被垃圾回收。

  2. 代码和环境是如何分离的?

    test 的代码逻辑静静地躺在堆内存里(右边),而它所需要操作的数据 timer 躺在动态生成的词法环境里(左边)。两者通过指针跨越内存区域相连。

所谓闭包,本质上就是:一个函数对象(代码)保留了对它出生地(词法环境)的引用。

理解了这一点,你就理解了 JavaScript 内存模型的核心。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/8 7:46:36

35、脚本编程与bash定制:邮件发送、流程自动化及提示定制全解析

脚本编程与bash定制:邮件发送、流程自动化及提示定制全解析 脚本中发送邮件 在脚本中实现邮件发送功能,有多种方式可供选择,但每种方式都有其特点和适用场景。 首先,使用uuencode方法发送附件时,邮件客户端的支持情况会影响附件的显示效果。像Thunderbird和Outlook这类…

作者头像 李华
网站建设 2026/6/10 9:12:22

safetensors 检查完整

python safetensors 检查是否完整:from safetensors import safe_opendef check_safetensors_basic(path):try:with safe_open(path, framework"pt") as f:keys list(f.keys())print("✅ safetensors 文件结构完整")print(f" tensor 数…

作者头像 李华
网站建设 2026/6/10 17:03:46

若依系统报错net::ERR_CONNECTION_TIMED_OUT的原因

一、错误核心含义net::ERR_CONNECTION_TIMED_OUT 是前端发起请求时,无法在指定时间内与后端服务器建立连接(连接超时),而非接口报错或返回异常。在若依(RuoYi)框架中,该错误几乎都与「网络连通性…

作者头像 李华
网站建设 2026/6/10 17:07:33

上海计算机学会12月月赛丙组T1强迫症题解

题目描述冰棍特别喜欢整齐的东西,对于数更是有特殊的癖好。冰棍认为,如果一个数只有恰好一位不是 00,那么这个数是整齐的。比如,1,2,40,5001,2,40,500 是整齐的数,而 1919,101,80801919,101,8080 不是整齐的数。现在他…

作者头像 李华
网站建设 2026/6/10 17:09:32

gemma.cpp模型转换实战:从Hugging Face到C++推理的高效路径

gemma.cpp模型转换实战:从Hugging Face到C推理的高效路径 【免费下载链接】gemma.cpp 适用于 Google Gemma 模型的轻量级独立 C 推理引擎。 项目地址: https://gitcode.com/GitHub_Trending/ge/gemma.cpp 你是否曾经在Python环境中训练了优秀的Gemma模型&…

作者头像 李华
网站建设 2026/6/10 17:10:06

26、高效办公必备:全面的键盘快捷键指南

高效办公必备:全面的键盘快捷键指南 在日常的文档处理、图形设计、表格操作等工作中,熟练掌握键盘快捷键可以极大地提高工作效率。以下将为大家详细介绍一系列实用的键盘快捷键及其使用方法。 1. 插入点移动 在文本编辑过程中,快速移动插入点是一项基本需求。以下是一些常…

作者头像 李华