词法作用域是闭包形成的底层前提,即函数定义时就确定其可访问的变量范围;闭包是嵌套函数引用外层变量并逃逸出其作用域后,因outer指针保持对外层词法环境的引用而自然形成的现象。

词法作用域是闭包形成的底层前提,不是运行时决定的,而是写代码时函数放在哪、嵌套在哪,就“锁定”了它能访问哪些变量。闭包不是刻意造出来的,是词法作用域 + 函数作为值被返回或传递时自然发生的现象。
词法作用域:定义时就定死的作用域范围
JavaScript 的作用域在函数声明那一刻就确定了,跟它后来在哪被调用完全无关。比如一个函数在全局定义,它的 outer 指针就指向全局词法环境;如果它定义在另一个函数内部,outer 就指向那个外层函数的词法环境。
这和“动态作用域”(比如某些 shell 脚本)完全不同——后者看的是调用链,而 JS 看的是代码结构本身。
函数里访问变量,先查自己词法环境(let/const)和变量环境(var) 找不到,就顺着 outer 指针往上找,一级一级直到全局 这个查找路径就是作用域链,它由 outer 指针串联而成,不是靠调用栈决定的
闭包形成:内部函数“带走”了外层变量的引用
当一个函数内部定义了另一个函数,并且这个内部函数引用了外部函数的局部变量(比如 let 声明的 count),同时这个内部函数以某种方式逃出了外部函数的作用域(比如作为返回值、赋给全局变量、传给 setTimeout),那么闭包就形成了。
立即学习“Java免费学习笔记(深入)”;
关键点在于:外部函数执行完后,它的执行上下文本该被销毁,但因为内部函数还“抓着”它的词法环境(通过 outer),JS 引擎就不会回收那些变量——它们被“活捉”住了。
必须有嵌套函数 内部函数必须引用外部函数的变量(哪怕只读) 内部函数必须在外部函数作用域之外被调用或保留
一个典型例子帮你串起来
看这段代码:
function makeCounter() {
let count = 0;
return function() {
return ++count;
};
}
const inc = makeCounter();
inc(); // 1
inc(); // 2
makeCounter 执行完后,count 理应消失,但它没消失——因为返回的那个匿名函数通过自己的 outer 指针,一直连着 makeCounter 创建时的词法环境。每次调用 inc,都是在操作那个被“封存”的 count。
闭包不是魔法,是词法作用域的必然结果
你不需要“制造”闭包,只要写了嵌套函数 + 引用了外层变量 + 让内层函数跑出去,闭包就自动存在。它本质是 JS 引擎为支持词法作用域而设计的内存管理机制:只要还有活跃的引用,变量就留着。
基本上就这些。
以上就是如何理解javascript词法作用域_闭包怎样形成?的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1543572.html
微信扫一扫
支付宝扫一扫