执行上下文与作用域

本文最后更新于:1 年前

执行上下文与作用域

执行上下文

执行上下文(以下简称“上下文”)的概念在 JavaScript 中是颇为重要的。变量或函数的上下文决定了它们可以访问哪些数据,以及它们的行为。每个上下文都有一个关联的变量对象(variable object),而这个上下文中定义的所有变量和函数都存在于这个对象上。
全局上下文是最外层的上下文。根据 ECMAScript 实现的宿主环境,表示全局上下文的对象可能不一样。在浏览器中,全局上下文就是我们常说的 window 对象),因此所有通过 var 定义的全局变量和函数都会成为 window 对象的属性和方法。
使用 let 和 const 的顶级声明不会定义在全局上下文中,但在作用域链解析上效果是一样的。上下文在其所有代码都执行完毕后会被销毁,包括定义在它上面的所有变量和函数(全局上下文在应用程序退出前才会被销毁,比如关闭网页或退出浏览器)。
每个函数调用都有自己的上下文。当代码执行流进入函数时,函数的上下文被推到一个上下文栈上。在函数执行完之后,上下文栈会弹出该函数上下文,将控制权返还给之前的执行上下文。ECMAScript 程序的执行流就是通过这个上下文栈进行控制的。
上下文中的代码在执行的时候,会创建变量对象的一个作用域链 (scope chain)。这个作用域链决定了各级上下文中的代码在访问变量和函数时的顺序。代码正在执行的上下文的变量对象始终位于作用域链的最前端。如果上下文是函数,则其活动对象(activation object)用作变量对象。
活动对象最初只有一个定义变量: arguments 。(全局上下文中没有这个变量)作用域链中的下一个变量对象来自包含上下文,再下一个对象来自再下一个包含上下文。以此类推直至全局上下文;全局上下文的变量对象始终是作用域链的最后一个变量对象。 代码执行时的标识符解析是通过沿作用域链逐级搜索标识符名称完成的。搜索过程始终从作用域链的最前端开始,然后逐级往后,直到找到标识符。(如果没有找到标识符,那么通常会报错。)

1
2
3
4
5
6
7
8
9
10
var color = 'blue'
function changeColor() {
if (color === 'blue') {
color = 'red'
} else {
color = 'blue'
}
}
changeColor()
console.log(color) // red

对这个例子而言,函数 changeColor() 的作用域链包含两个对象:一个是它自己的变量对象(就是定义 arguments 对象的那个),另一个是全局上下文的变量对象。这个函数内部之所以能够访问变量 color ,就是因为可以在作用域链中找到它。
此外,局部作用域中定义的变量可用于在局部上下文中替换全局变量。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
var color = 'blue'
function changeColor() {
let anotherColor = 'red'
function swapColors() {
let tempColor = anotherColor
anotherColor = color
color = tempColor
// 这里可以访问color、anotherColor和tempColor
console.log(1, color, anotherColor, tempColor) // 1 red blue red
}
// 这里可以访问color和anotherColor,但访问不到tempColor
swapColors()
console.log(2, color, anotherColor) // 2 red blue
}
// 这里只能访问color
changeColor()
console.log(3, color) // 3 red
// 1 red blue red
// 2 red blue
// 3 red

以上代码涉及 3 个上下文:全局上下文changeColor() 的局部上下文swapColors() 的局部上下文
全局上下文中有一个变量 color 和一个函数 chageColor() 。
changeColor() 的局部上下文中有一个变量 anotherColor 和一个函数 swapColors() ,但在这里可以访问全局上下文中的变量 color 。
swapColors() 的局部上下文中有一个变量 tempColor ,只能在这个上下文中访问到。全局上下文和 changeColor() 的局部上下文都无法访问到 tempColor 。而在 swapColors() 中则可以访问另外两个上下文中的变量,因为它们都是父上下文。

内部上下文可以通过作用域链访问外部上下文中的一切,但外部上下文无法访问内部上下文中的任何东西。上下文之间的连接是线性的、有序的。每个上下文都可以到上一级上下文中去搜索变量和函数,但任何上下文都不能到下一级上下文中去搜索。
swapColors() 局部上下文的作用域链中有 3 个对 象: swapColors() 的变量对象、 changeColor() 的变量对象 和全局变量对象。 swapColors() 的局部上下文首先从自己的变量对象开始搜索变量和函数,搜不到就去搜索上一级变量对象。
changeColor() 上下文的作用域链中只有 2 个对象:它自己的变量对象和全局变量对象。因此,它不能访问 swapColors() 的上下 文。

作用域链增强

虽然执行上下文主要有全局上下文和函数上下文两种( eval() 调用内部存在第三种上下文),但有其他方式来增强作用域链。某些语句会导致在作用域链前端临时添加一个上下文,这个上下文在代码 执行后会被删除。
通常在两种情况下会出现这个现象,即代码执行到下面任意一种情况时:

  • try / catch 语句的 catch 块
  • with 语句

这两种情况下,都会在作用域链前端添加一个变量对象。对 with 语句来说,会向作用域链前端添加指定的对象;对 catch 语句而言,则会创建一个新的变量对象,这个变量对象会包含要抛出的错误对象的声明

1
2
3
4
5
6
7
function buildUrl() {
let qs = '?debug=true'
with (location) {
let url = href + qs
}
return url
}

这里, with 语句将 location 对象作为上下文,因此 location 会被添加到作用域链前端。 buildUrl() 函数中定义了一个变量 qs 。当 with 语句中的代码引用变量 href 时,实际上引用的是 location.href ,也就是自己变量对象的属性。在引用 qs 时,引用的则是定义在 buildUrl() 中的那个变量,它定义在函数上下文的变量对象上。而在 with 语句中使用 var 声明的变量 url 会成为函数上下文的一部分,可以作为函数的值被返回;但像这里使用 let 声明的变量 url ,因为被限制在块级作用域(稍后介 绍),所以在 with 块之外没有定义

变量声明

直到 ECMAScript 5.1, var 都是声明变量的唯一关键字。ES6 不仅增加了 let 和 const 两个关键字,而且还让这两个关键字压倒性地超越 var 成为首选。

使用 var 的函数作用域声明

在使用 var 声明变量时,变量会被自动添加到最接近的上下文。在函数中,最接近的上下文就是函数的局部上下文。在 with 语句中,最接近的上下文也是函数上下文。如果变量未经声明就被初始化了,那么它就会自动被添加到全局上下文

1
2
3
4
5
6
function add(num1, num2) {
var sum = num1 + num2
return sum
}
let result = add(10, 20) // 30
console.log(sum) // 报错:sum在这里不是有效变量

这里,函数 add() 定义了一个局部变量 sum ,保存加法操作的结果。这个值作为函数的值被返回,但变量 sum 在函数外部是访问不到的。如果省略上面例子中的关键字 var ,那么 sum 在 add() 被调用之后就变成可以访问的了

1
2
3
4
5
6
function add(num1, num2) {
sum = num1 + num2
return sum
}
let result = add(10, 20) // 30
console.log(sum)

这一次,变量 sum 被用加法操作的结果初始化时并没有使用 var 声明。在调用 add() 之后, sum 被添加到了全局上下文,在函数退出之后依然存在,从而在后面可以访问到。

注意 未经声明而初始化变量是 JavaScript 编程中一个非常常见的错误,会导致很多问题。

var 声明会被拿到函数或全局作用域的顶部,位于作用域中所有代码之前。这个现象叫作“提升”(hoisting)。提升让同一作用域中的代码不必考虑变量是否已经声明就可以直接使用。

1
2
3
4
var name = 'Jake'
// 等价于:
name = 'Jake'
var name
1
2
3
4
5
6
7
8
function fn1() {
var name = 'Jake'
}
// 等价于:
function fn2() {
var name
name = 'Jake'
}

通过在声明之前打印变量,可以验证变量会被提升。声明的提升意味着会输出 undefined 而不是 Reference Error

1
2
3
4
5
6
7
8
console.log(name1) // undefined
var name1 = 'Jake'

function a1() {
console.log(name2) // undefined
var name2 = 'Jake'
}
a1()

使用 let 的块级作用域声明

ES6 新增的 let 关键字跟 var 很相似,但它的作用域是块级的,这也是 JavaScript 中的新概念。块级作用域由最近的一对包含花括号 {} 界定。换句话说, if 块、 while 块、 function 块,甚至连单独的块也是 let 声明变量的作用域

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
if (true) {
let a
}
console.log(a) // ReferenceError: a没有定义
while (true) {
let b
}
console.log(b) // ReferenceError: b没有定义
function foo() {
let c
}
console.log(c) // ReferenceError: c没有定义
// 这没什么可奇怪的
// var声明也会导致报错
// 这不是对象字面量,而是一个独立的块
// JavaScript解释器会根据其中内容识别出它来
{
let d
}
console.log(d) // ReferenceError: d没有定义

let 与 var 的另一个不同之处是在同一作用域内不能声明两次。重复的 var 声明会被忽略,而重复的 let 声明会抛出 SyntaxError 。

1
2
3
4
5
6
7
8
var a;
var a;
// 不会报错
{
let b;
let b;
}
// SyntaxError: 标识符b已经声明过

let 的行为非常适合在循环中声明迭代变量。使用 var 声明的迭代变量会泄漏到循环外部,这种情况应该避免。

1
2
3
4
for (var i = 0; i < 10; ++i) {}
console.log(i); // 10
for (let j = 0; j < 10; ++j) {}
console.log(j); // ReferenceError: j没有定义

严格来讲, let 在 JavaScript 运行时中也会被提升,但由于“暂时性死区”(temporal dead zone)的缘故,实际上不能在声明之前使用 let 变量。因此,从写 JavaScript 代码的角度说, let 的提升跟 var 是不一样的。

使用 const 的常量声明

除了 let ,ES6 同时还增加了 const 关键字。使用 const 声明的变量必须同时初始化为某个值。一经声明,在其生命周期的任何时候都不能再重新赋予新值

1
2
3
4
const a; // SyntaxError: 常量声明时没有初始化
const b = 3;
console.log(b); // 3
b = 4; // TypeError: 给常量赋值

const 除了要遵循以上规则,其他方面与 let 声明是一样的
const 声明只应用到顶级原语或者对象。换句话说,**赋值为对象的 const 变量不能再被重新赋值为其他引用值,但对象的键则不受限制。 **

1
2
3
4
5
const o1 = {};
o1 = {}; // TypeError: 给常量赋值
const o2 = {};
o2.name = 'Jake';
console.log(o2.name); // 'Jake'

如果想让整个对象都不能修改,可以使用 Object.freeze() ,这样再给属性赋值时虽然不会报错,但会静默失败:

1
2
3
const o3 = Object.freeze({})
o3.name = 'Jake'
console.log(o3.name) // undefined

Object.freeze() 方法可以冻结一个对象。一个被冻结的对象再也不能被修改;冻结了一个对象则不能向这个对象添加新的属性,不能删除已有属性,不能修改该对象已有属性的可枚举性、可配置性、可写性,以及不能修改已有属性的值。此外,冻结一个对象后该对象的原型也不能被修改。freeze() 返回和传入的参数相同的对象。

1
2
3
4
5
6
7
8
const obj = {
prop: 42
};
Object.freeze(obj);
obj.prop = 33;
// Throws an error in strict mode
console.log(obj.prop);
// expected output: 42

总结

任何变量(不管包含的是原始值还是引用值)都存在于某个执行上下文中(也称为作用域)。这个上下文(作用域)决定了变量的生命周期,以及它们可以访问代码的哪些部分。

  • 执行上下文分全局上下文、函数上下文和块级上下文。
  • 代码执行流每进入一个新上下文,都会创建一个作用域链,用于搜索变量和函数。
  • 函数或块的局部上下文不仅可以访问自己作用域内的变量,而且也可以访问任何包含上下文乃至全局上下文中的变量。
  • 全局上下文只能访问全局上下文中的变量和函数,不能直接访问局部上下文中的任何数据。
  • 变量的执行上下文用于确定什么时候释放内存。