前端攻城狮查缺补漏系列-执行上下文、作用域、作用域链

108 阅读4分钟

执行上下文

执行上下文是指javascript运行环境的一种抽象概念,javascript代码再运行时实际上就是在执行上下文中运行的。

执行上下文类型

1:全局执行上下文

任何不在函数内部的代码都在全局上下文中。它会执行两件事:创建一个全局的 window 对象(浏览器的情况下),并且设置 this 的值等于这个全局对象。一个程序中只会有一个全局执行上下文

2: 函数执行上下文

每当一个函数 被调用时, 都会为该函数创建一个新的上下文。每个函数都有它自己的执行上下文,不过是在函数被调用时创建的。函数上下文可以有任意多个。每当一个新的执行上下文被创建,它会按定义的顺序(将在后文讨论)执行一系列步骤。

3: Eval 函数执行上下文

执行在 eval 函数内部的代码也会有它属于自己的执行上下文(不常用,这里不考虑)

执行栈

执行栈,也就是在其它编程语言中所说的“调用栈”,被用来存储代码运行时创建的所有执行上下文。

代码执行顺序

代码演示

let a = 'Hello World!';
function first() {
  console.log('Inside first function');
  second();
  console.log('Again inside first function');
}
function second() {
  console.log('Inside second function');
}
first();
console.log('Inside Global Execution Context');

tupian

当上述代码在浏览器加载时,JavaScript 引擎创建了一个全局执行上下文并把它压入当前执行栈。当遇到 first() 函数 调用时,JavaScript 引擎为该函数创建一个新的执行上下文并把它压入当前执行栈的顶部。当从 first() 函数内部调用 second() 函数时,JavaScript 引擎为 second() 函数创建了一个新的执行上下文并把它压入当前执行栈的顶部。当 second() 函数执行完毕,它的执行上下文会从当前栈弹出,并且控制流程到达下一个执行上下文,即 first() 函数的执行上下文。

作用域

1:什么事作用域

可访问变量的区域,换句话说,作用域决定了代码区块中变量和其他资源的可见性。

2:全局作用域和函数作用域

在代码中任何地方都能访问到的对象拥有全局作用域,在函数体内申明的变量,函数外无法访问,这是函数作用域,代码演示

var outVariable = "我是最外层变量";
function outFun() { //最外层函数
    var inVariable = "内层变量";
    function innerFun() { //内层函数
        console.log(inVariable);
    }
    innerFun();
}
console.log(outVariable); //我是最外层变量
outFun(); //内层变量
console.log(inVariable); //inVariable is not defined
innerFun(); //innerFun is not defined

3:块级作用域

块级作用域可通过ES6新增命令 let 和 const 声明,所声明的变量在指定块的作用域外无法被访问。块级作用域在如下情况被创建:

1:在一个函数内部

function outFun() { 
    var inVariable = "内层变量";
}
console.log(inVariable) // msc is not defined

2:在一个代码块(由一对花括号包裹)内部

{
  let msc = '马士春'
}
console.log(msc) // msc is not defined

作用域链

1:什么事自由变量

var a = 100
function fn() {
    var b = 200
    console.log(a) // 这里的a在这里就是一个自由变量
    console.log(b)
}
fn()

当前作用域没有定义的变量,这就是自由变量 。自由变量的值如何得到 —— 向父级作用域寻找(注意:这种说法并不严谨,下文会重点解释)。

2:什么是作用域

如果父级也没呢?再一层一层向上寻找,直到找到全局作用域还是没找到,就宣布放弃。这种一层一层的关系,就是 作用域链 。

var a = 100
function F1() {
    var b = 200
    function F2() {
        var c = 300
        console.log(a) // 自由变量,顺作用域链向父作用域找
        console.log(b) // 自由变量,顺作用域链向父作用域找
        console.log(c) // 本作用域的变量
    }
    F2()
}
F1()

3:关于自由变量的取值

关于自由变量的值,上文提到要到父作用域中取,其实有时候这种解释会产生歧义.

var x = 10
function fn() {
  console.log(x)
}
function show(f) {
  var x = 20
  (function() {
    f() //10,而不是20
  })()
}
show(fn)

在 fn 函数中,取自由变量 x 的值时,要到哪个作用域中取?——要到创建 fn 函数的那个作用域中取,无论 fn 函数将在哪里调用。所以,不要在用以上说法了。相比而言,用这句话描述会更加贴切:**要到创建这个函数的那个域”。 重点重点重点,(还记得this指向那节吗?this的取值刚好和作用域相反,this是在函数被调用时创建的。)