news 2026/5/17 10:53:31

闭包里的变量到底存哪了?图解 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/5/12 23:46:16

25、文档管理与超文本应用全解析

文档管理与超文本应用全解析 在文档处理过程中,我们常常会遇到各种复杂的情况,比如条件文本的管理、文档更改的跟踪以及超文本的应用等。下面将详细介绍这些方面的相关知识和操作方法。 条件文本管理 条件文本在概念上相对容易理解,但在复杂文档中管理起来却颇具挑战。如…

作者头像 李华
网站建设 2026/5/16 6:05:08

Actix Web终极实战指南:从零构建高性能Rust微服务

Actix Web终极实战指南:从零构建高性能Rust微服务 【免费下载链接】actix-web Actix Web is a powerful, pragmatic, and extremely fast web framework for Rust. 项目地址: https://gitcode.com/gh_mirrors/ac/actix-web 你是否正在寻找一个既能提供极致性…

作者头像 李华
网站建设 2026/5/17 8:17:46

基于STM32的平衡车开发计划-1-PWM设置

我们使用的是STM32F103C8T6进行编程和实现。一.配置一下PWM输出1.PWM设置HAL_TIM_PWM_Start(&htim1,TIM_CHANNEL_1);启动PWM使用的是TIM1的ch1通道,也就是PA8端口,cubemx会自动帮我们配置好。预分频为72,自从重装值1000,采用向…

作者头像 李华
网站建设 2026/5/12 23:41:40

在STM32函数指针是什么,怎么使用还有典型应用场景。

函数指针听起来复杂,但其实你可以把它理解成一个“遥控器”。它本身不干活,但按一下(调用它),就能遥控执行另一个函数。在STM32开发里,这个“遥控”的特性,正好能解决硬件和应用之间灵活联动的问…

作者头像 李华
网站建设 2026/5/14 1:40:29

进销存最新推荐 | 哪个进销存软件性价比最好?- 象过河软件

在数字化管理工具选型中,进销存软件的 “性价比” 从来不是单纯看价格,而是 “功能实用性 投入成本” 的综合考量。市面上有的软件低价但功能残缺,有的功能齐全却年费高昂,还有的隐藏插件收费,让企业陷入 “两难”。2…

作者头像 李华
网站建设 2026/5/12 5:12:46

5个关键技巧:高效使用baseimage-docker构建企业级容器环境

5个关键技巧:高效使用baseimage-docker构建企业级容器环境 【免费下载链接】baseimage-docker A minimal Ubuntu base image modified for Docker-friendliness 项目地址: https://gitcode.com/gh_mirrors/ba/baseimage-docker 在容器化技术日益普及的今天&a…

作者头像 李华