本文重点是要梳理执行上下文的生命周期中的建立作用域链,在此之前,先回顾下关于作用域的一些知识。
1.什么是作用域(scope)?
在《JavaScritp高级程序设计》中并没有找到确切的关于作用域的定义,只是在“4.2执行环境及作用域”中简单说了下执行环境(execution context)的概念。而执行环境其实就是之前博客:js基础梳理-究竟什么是执行上下文栈(执行栈),执行上下文(可执行代码)?中的执行上下文。
而在《JavaScript权威指南》中,对作用域的描述为:
变量作用域:一个变量的作用域(scope)是程序源代码中定义这个变量的区域
在《你不知道的Javascript·上卷》中对作用域的描述则为:
负责收集并维护由所有生命的标识符(变量)组成的一系列查询,并实施一套非常严格的规则,确定当前执行的代码对这些标识符的访问权限。
简单来讲,作用域(scope)就是变量访问规则的有效范围。
- 作用域外,无法引用作用域内的变量;
- 离开作用域后,作用域的变量的内存空间会被清除,比如执行完函数或者关闭浏览器
- 作用域与执行上下文是完全不同的两个概念。我曾经也混淆过他们,但是一定要仔细区分。
JavaScript代码的整个执行过程,分为两个阶段,代码编译阶段与代码执行阶段。编译阶段由编译器完成,将代码翻译成可执行代码,这个阶段作用域规则会确定。执行阶段由引擎完成,主要任务是执行可执行代码,执行上下文在这个阶段创建。
说得很深奥的样子,其实上面这段话重点用函数作用域与函数执行上下文来区分是最好不过的了。函数作用域是在函数声明的时候就已经确定了,而函数执行上下文是在函数调用时创建的。假如一个函数被调用多次,那么它就会创建多个函数执行上下文,但是函数作用域显然不会跟着函数被调用的次数而发生什么变化。
1.1 全局作用域
var foo = 'foo'; console.log(window.foo); // => 'foo' 在浏览器环境中声明变量,该变量会默认成为window对象下的属性。
function foo() { name = "bar" } foo(); console.log(window.name) // bar在函数中,如果不加 var 声明一个变量,那么这个变量会默认被声明为全局变量,如果是严格模式,则会报错。
全局变量会造成命名污染,如果在多处对同一个全局变量进行操作,那么久会覆盖全局变量的定义。同时全局变量数量过多,非常不方便管理。
这也是为什么jquery要在全局建立变量
