JavaScript执行上下文分全局、函数、eval三种,分别在页面加载、函数调用、eval执行时创建;作用域链在函数定义时静态确定,沿词法环境逐级向上查找变量,未命中则抛ReferenceError。
JavaScript 的执行上下文(Execution Context)和作用域链(Scope Chain)不是两个独立概念,而
是运行时机制的一体两面:执行上下文定义了代码执行的“环境容器”,作用域链则是该容器内查找变量时实际走的那条路径。
JS 引擎每次执行一段可执行代码(global code、function code、eval code)前,都会先创建一个执行上下文。它分为三类:
Global Execution Context:页面加载时自动创建,对应 window(浏览器)或 globalThis(Node.js),this 指向全局对象Function Execution Context:每次调用函数时创建,包含自己的 arguments、this 和词法环境(LexicalEnvironment)Eval Execution Context:极少用,且严格模式下禁用,不建议关注注意:setTimeout 回调、事件处理函数、Promise 回调等异步代码,执行时也各自进入新的函数执行上下文,不是“延续”上一个上下文。
ReferenceError
作用域链是执行上下文内部的一个只读属性([[Scopes]]),由当前函数的词法环境向上逐级链接到外层词法环境,最终到全局环境。它在函数定义时就确定(静态),而非调用时决定。
立即学习“Java免费学习笔记(深入)”;
变量查找过程就是沿着这条链逐级搜索:
LexicalEnvironment(含 let/const 声明)VariableEnvironment(含 var 声明)LexicalEnvironment,直到全局ReferenceError(不是 undefined)function outer() {
const x = 10;
function inner() {
console.log(x); // 沿作用域链找到 outer 的 LexicalEnvironment 中的 x
}
inner();
}
outer();
var 和 let/const 在作用域链中表现不同?不是作用域链本身不同,而是它们绑定的位置不同:
var 声明被提升并绑定到当前执行上下文的 VariableEnvironment,所以能在声明前访问(值为 undefined)let/const 绑定到 LexicalEnvironment,但存在“暂时性死区”(TDZ):从块顶部到声明语句之间,访问会直接抛 ReferenceError
console.log(a); // undefined(var 提升) console.log(b); // ReferenceError(TDZ) var a = 1; let b = 2;
闭包本质是函数对象持有了对其定义时所在词法环境的引用。即使外层函数执行结束、其执行上下文本该被销毁,只要内层函数还存在(比如被返回、被赋值给全局变量),JS 引擎就会保留该外层环境——因为作用域链仍需要它。
容易忽略的关键点:
let 声明 i,所有回调共享同一个 i)真正难理解的,从来不是“链怎么连”,而是“哪些环境被保留、哪些被释放”——这取决于引擎的垃圾回收策略与你是否无意中维持了对词法环境的强引用。