Jamey's Jamey's
首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于

Jamey

首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于
  • 深入系列

    • 为什么要写?
    • JavaScript 深入之从原型到原型链
    • JavaScript 深入之词法作用域和动态作用域
    • JavaScript 深入之执行上下文栈
      • 一. 顺序执行?
      • 二. 可执行代码
      • 三. 执行上下文栈
      • 四. 解答思考题
    • JavaScript 深入之变量对象
    • JavaScript 深入之作用域链
    • JavaScript 深入之从 ECMAScript 规范解读 this
    • JavaScript 深入之执行上下文
    • JavaScript 深入之闭包
    • JavaScript 深入之参数按值传递
    • JavaScript 深入之 call 和 apply 的模拟实现
    • JavaScript 深入之 bind 的模拟实现
    • JavaScript 深入系列之 new 的模拟实现
    • JavaScript 深入之类数组对象与 arguments
    • JavaScript 深入之创建对象的多种方式以及优缺点
    • JavaScript 深入之继承的多种方式以及优缺点
    • JavaScript 深入之浮点数精度
    • JavaScript 深入之头疼的类型转换(上)
    • JavaScript 深入之头疼的类型转换(下)
  • 专题系列

  • underscore系列

  • ES6系列

  • 模块化

  • 正则表达式

  • 单元测试

  • 微前端

  • 实用函数

  • Rollup

  • 解决方案

  • 《JavaScript》笔记
  • 深入系列
Jamey
2021-10-03
目录

JavaScript 深入之执行上下文栈

# JavaScript 深入之执行上下文栈

# 一. 顺序执行?

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

var foo = function() {
  console.log('foo1');
}

foo(); // foo1

var foo = function() {
  console.log('foo2');
}

foo(); // foo2
1
2
3
4
5
6
7
8
9
10
11

然而去看这段代码:

function foo() {
  console.log('foo1');
}

foo(); // foo2

function foo() {
  console.log('foo2');
}

foo(); // foo2
1
2
3
4
5
6
7
8
9
10
11

打印的结果却是两个 foo2。

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

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

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

# 二. 可执行代码

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

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

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

# 三. 执行上下文栈

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

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

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

ECStack = [];
1

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

ECStack = [
  globalContext
];
1
2
3

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

function fun3() {
  console.log('fun3');
}

function fun2() {
  fun3();
}

function fun1() {
  fun2();
}

fun1();
1
2
3
4
5
6
7
8
9
10
11
12
13

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

// 伪代码

// 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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

# 四. 解答思考题

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

var scope = "global scope";
function checkscope(){
    var scope = "local scope";
    function f(){
        return scope;
    }
    return f();
}
checkscope();
1
2
3
4
5
6
7
8
9
var scope = "global scope";
function checkscope(){
    var scope = "local scope";
    function f(){
        return scope;
    }
    return f;
}
checkscope()();
1
2
3
4
5
6
7
8
9

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

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

让我们模拟第一段代码:

ECStack.push(<checkscope> functionContext);
ECStack.push(<f> functionContext);
ECStack.pop();
ECStack.pop();
1
2
3
4

让我们模拟第二段代码:

ECStack.push(<checkscope> functionContext);
ECStack.pop();
ECStack.push(<f> functionContext);
ECStack.pop();
1
2
3
4

是不是有些不同呢?

当然了,这样概括的回答执行上下文栈的变化不同,是不是依然有一种意犹未尽的感觉呢。为了更详细讲解两个函数执行上的区别,我们需要探究一下执行上下文到底包含了哪些内容,欢迎阅读下一篇《JavaScript深入之变量对象》

#JavaScript 深入
上次更新: 2022/07/01, 17:34:19
JavaScript 深入之词法作用域和动态作用域
JavaScript 深入之变量对象

← JavaScript 深入之词法作用域和动态作用域 JavaScript 深入之变量对象→

Theme by Vdoing | Copyright © 2017-2023 Jamey | blog 闽ICP备19022664号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式