查看原文
其他

JavaScript 深入之执行上下文栈

2017-05-15 前端大全

(点击上方公众号,可快速关注)


作者:冴羽

github.com/mqyqingfeng/Blog/issues/4

如有好文章投稿,请点击 → 这里了解详情


顺序执行?


如果要问到JavaScript代码执行顺序的话,想必写过JavaScript的开发者都会有个直观的印象,那就是顺序执行,毕竟


var foo = function () {

 

    console.log('foo1');

 

}

 

foo();  // foo1

 

var foo = function () {

 

    console.log('foo2');

 

}

 

foo(); // foo2


然而去看这段代码:


function foo() {

 

    console.log('foo1');

 

}

 

foo();  // foo2

 

function foo() {

 

    console.log('foo2');

 

}

 

foo(); // foo2


打印的结果却是两个foo2。


刷过面试题的都知道这是因为JavaScript引擎并非一行一行地分析和执行程序,而是一段一段地分析执行。当执行一段代码的时候,会进行一个“准备工作”,比如第一个例子中的变量提升,和第二个例子中的函数提升。


但是本文真正想让大家思考的是:这个”一段一段”中的“段”究竟是怎么划分的呢?


到底JavaScript引擎遇到一段怎样的代码时才会做’准备工作’呢?


可执行代码


这就要说到JavaScript的可执行代码(executable code)的类型有哪些了?


其实很简单,就三种,全局代码、函数代码、eval代码。


举个例子,当执行到一个函数的时候,就会进行准备工作,这里的’准备工作’,让我们用个更专业一点的说法,就叫做”执行上下文(execution contexts)”。


执行上下文栈


接下来问题来了,我们写的函数多了去了,如何管理创建的那么多执行上下文呢?


所以js引擎创建了执行上下文栈(Execution context stack,ECS)来管理执行上下文


为了模拟执行上下文栈的行为,让我们定义执行上下文栈是一个数组:


ECStack = [];


试想当JavaScript开始要解释执行代码的时候,最先遇到的就是全局代码,所以初始化的时候首先就会向执行上下文栈压入一个全局执行上下文,让我们用globalContext表示它,并且只有当整个应用程序结束的时候,ECStack才会被清空,所以ECStack最底部永远有个globalContext:


    ECStack = [

        globalContext

    ];


现在JavaScript遇到下面的这段代码了:


function fun3() {

    console.log('fun3')

}

 

function fun2() {

    fun3();

}

 

function fun1() {

    fun2();

}

 

fun1();


当遇到函数执行的时候,就会创建一个执行上下文,并且压入执行上下文栈,当函数执行完毕的时候,就会将函数的执行上下文从栈中弹出。知道了这样的工作原理,让我们来看看如何处理上面这段代码:


// 伪代码

 

// fun1()

ECStack.push(fun1> functionContext);

 

// fun1中竟然调用了fun2,还要创建fun2的执行上下文

ECStack.push(fun2> functionContext);

 

// 擦,fun2还调用了fun3!

ECStack.push(fun3> functionContext);

 

// fun3执行完毕

ECStack.pop();

 

// fun2执行完毕

ECStack.pop();

 

// fun1执行完毕

ECStack.pop();

 

// javascript接着执行下面的代码,但是ECStack底层用于有个globalContext

 

解答思考题


好啦,到此为止,我们已经了解了执行上下文栈如何处理执行上下文的,所以让我们看看《JavaScript深入之词法作用域和动态作用域》这篇文章最后的问题:


var scope = "global scope";

function checkscope(){

    var scope = "local scope";

    function f(){

        return scope;

    }

    return f();

}

checkscope();


var scope = "global scope";

function checkscope(){

    var scope = "local scope";

    function f(){

        return scope;

    }

    return f;

}

checkscope()();


两段代码执行的结果一样,但是两段代码究竟有哪些不同呢?


答案就是执行上下文栈的变化不一样。


让我们模拟第一段代码:


ECStack.push(checkscope> functionContext);

ECStack.push(f> functionContext);

ECStack.pop();

ECStack.pop();


让我们模拟第二段代码:


ECStack.push(checkscope> functionContext);

ECStack.pop();

ECStack.push(f> functionContext);

ECStack.pop();


是不是有些不同呢?


当然,如果觉得这样粗略的回答执行上下文栈的变化,依然显得不够详细,那就让我们去探究一下执行上下文到底包含了哪些内容,欢迎期待下一篇《JavaScript深入之变量对象》


深入系列


JavaScript深入系列预计写十五篇左右,旨在帮大家捋顺JavaScript底层知识,重点讲解如原型、作用域、执行上下文、变量对象、this、闭包、按值传递、call、apply、bind、new、继承等难点概念,与罗列它们的用法不同,这个系列更注重通过写demo,捋过程、模拟实现,结合ES规范等方法来讲解。


所有文章和demo都可以在github上https://github.com/mqyqingfeng/Blog找到。如果有错误或者不严谨的地方,请务必给予指正,十分感谢。如果喜欢或者有所启发,欢迎star,对作者也是一种鼓励。


本系列:


  1. JavaScirpt 深入之从原型到原型链

  2. JavaScript 深入之词法作用域和动态作用域



觉得本文对你有帮助?请分享给更多人

关注「前端大全」,提升前端技能

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存