详解JavaScript中的执行上下文及调用堆栈


Posted in Javascript onApril 29, 2021

一、执行上下文是什么

代码运行是在一定的环境之中运行的,这个运行环境我们就成为执行环境,也就是执行上下文,按照执行环境不同,我们可以分为三类:

全局执行环境:代码首次执行时候的默认环境

函数执行环境:每当执行流程进入到一个函数体内部的时候

Eval执行环境:当eval函数内部的文本执行的时候

二、执行上下文栈是什么

既然是‘栈',那就得符合‘栈'的特性,即数据结构是先进后出。下面我们看一段代码:

function cat(a){
    if(a<0){
        return false;
    }
    console.log('入栈:'+a);
    cat(a-1);
    console.log('出栈:'+a);
}
cat(3);
// 入栈:3
// 入栈:2
// 入栈:1
// 入栈:0
// 出栈:0
// 出栈:1
// 出栈:2
// 出栈:3

我们来分析上面的代码的执行过程:

①浏览器加载时,程序进入全局执行上下文,将其压入一个栈内(第一个进入的,所以最底层);该执行上下文下只有一个函数cat、cat调用、参数3;

②程序进入cat函数内,即进入该函数执行上下文,也将其压入栈内(第二个进入的,所以倒数第二层),因为参数为3大于0,继续往下执行,输出'入栈:3';

③程序继续执行,调用函数cat,再次进入新的函数执行上下文,继续压入栈内(第三个进入,倒数第三层),参数为a-1,循环②步骤;这里,需要注意的是,因为调用了函数cat(a-1),导致下一行代码'出栈:a'(此时a仍为3),也就是'出栈:3'暂时搁浅起来,存在栈内倒数第二层

④不断重复②③步骤,以次输出'入栈:2'、'入栈:1'、'入栈:0';同时被搁浅的有'出栈:2'(栈内倒数第三层)、'出栈:1(栈内倒数第四层)'、'出栈:0(栈内倒数第五层)';

⑤按照栈的特性,被搁浅起来的4个输出项:以次输出'入栈:3'、'入栈:2'、'入栈:1'、'入栈:0';
以上就是执行上下文栈的具体情况,请大家手动代码练习一下,相信会容易理解透彻。

详解JavaScript中的执行上下文及调用堆栈

三、执行上下文栈的过程细节

我们已经知道,每次调用函数都会执行一个新的上下文,每个执行上下文都分为两个阶段:创建阶段、执行阶段
创建阶段:指的是程序调用函数,但代码未执行时的阶段;
执行阶段:指的是变量分配、函数执行等代码执行阶段;

(一)创建阶段

该阶段会调用函数数时,创建一个执行上下文对象(executionContextObj),该对象又包含了三个对象,分别是作用域链对象(scopeChain)、变量对象(variableObject,简称VO)、this对象,其中VO包括变量声明(variable)、函数声明(function)、参数(arguments)等。
这三个对象分别是在三个步骤创建的:
步骤1:初始化作用域链(scopeChain),开辟栈内存

步骤2:创建参数、函数、变量

步骤3:决定上下文的'this'的值

结合代码来进一步分析一下上面的步骤:

function cat(name) {
    var a = '年年';
    var b = function () {};
    this.name = name;
    function c() {
        console.log(this.name);
    }
    c();
}
cat('有鱼');

这段代码在调用函数 cat('有鱼')时,执行上下文是处于 创建阶段的,代码解析为:

cat执行上下文对象 = {
    scopeChain: { ... },// 1.创建作用域链,开辟栈内存
    variableObject: { 2.创建变量对象
        arguments: {  // 2.1 解析参数
            0: '有鱼',
            length: 1
        },
        name: '有鱼',  // 2.1 解析参数,创建形参名称,并进行参数赋值
        c: function c()  // 2.2 找到函数声明c,并将c作为属性,function c作为值
        a: undefined,  // 2.3 找到变量声明a,初始化为undefined,该阶段只看声明部分,不进行赋值
        b: undefined   // 2.3 找到变量声明b,初始化为undefined,该阶段只看声明部分,不进行赋值
    },
    this: {  3.创建this对象
        this:cat('有鱼') // 3.1 指向此次调用的对象
        name:undefined // 3.2 对象属性name的初始化为undefined
    };
    c() //又进入函数c执行上下文,跟cat函数一样,暂不展开 
}

通过代码解析我们可以得出以下结论

①三个步骤顺序不能乱
②VO步骤内,先执行函数声明,再执行变量声明
③只有参数可以在此阶段进行赋值,变量、函数都只能声明

详解JavaScript中的执行上下文及调用堆栈

(二)执行阶段

该阶段js解释器执行上下文中的函数代码,逐行运行js代码,并给变量赋值;
还是结合代码来分析:

cat执行上下文对象 = {
    scopeChain: { ... },
    variableObject: { 
        arguments: { 
            0: '有鱼',
            length: 1
        },
        name: '有鱼', 
        c: function c(),
        a: '年年',  // 变量a进行赋值
        b: function b   // 变量b进行赋值
    },
    this: {  3.创建this对象
        this:cat('有鱼') 
        name:'有鱼' // 对象属性name进行赋值
    }
}

以上就是详解JavaScript中的执行上下文及调用堆栈的详细内容,更多关于JavaScript中的执行上下文及调用堆栈的资料请关注三水点靠木其它相关文章!

Javascript 相关文章推荐
js 自制滚动条的小例子
Mar 16 Javascript
JavaScript获取/更改文本框的值的实例代码
Aug 02 Javascript
js实现简单的星级选择器提交效果适用于评论等
Oct 18 Javascript
JQuery异步加载无限下拉框级联功能实现示例
Feb 19 Javascript
JavaScript获取文本框内选中文本的方法
Feb 20 Javascript
jQuery使用addClass()方法给元素添加多个class样式
Mar 26 Javascript
JS实现来回出现文字的状态栏特效代码
Oct 31 Javascript
jQuery 的 ready()的纯js替代方法
Nov 20 Javascript
js实现一键复制功能
Mar 16 Javascript
AngularJS service之select下拉菜单效果
Jul 28 Javascript
Koa项目搭建过程详细记录
Apr 12 Javascript
vue-cli脚手架搭建的项目去除eslint验证的方法
Sep 29 Javascript
JavaScript实现淘宝商品图切换效果
JavaScript实现显示和隐藏图片
Apr 29 #Javascript
JS Canvas接口和动画效果大全
Apr 29 #Javascript
JS ES6异步解决方案
Apr 29 #Javascript
聊聊JS ES6中的解构
Apr 29 #Javascript
详解TypeScript中的类型保护
Apr 29 #Javascript
7个你应该知道的JS原生错误类型
Apr 29 #Javascript
You might like
PHP常用代码大全(新手入门必备)
2010/06/29 PHP
php 无限级分类,超级简单的无限级分类,支持输出树状图
2014/06/29 PHP
php遍历数组的4种方法总结
2014/07/05 PHP
详解php实现页面静态化原理
2017/06/21 PHP
PHP结合Redis+MySQL实现冷热数据交换应用案例详解
2019/07/09 PHP
使用JavaScript动态设置样式实现代码(2)
2013/01/25 Javascript
jquery如何把参数列严格转换成数组实现思路
2013/04/01 Javascript
jquery获得option的值和对option进行操作
2013/12/13 Javascript
JS图像无缝滚动脚本非常好用
2014/02/10 Javascript
JavaScript中使用arguments获得函数传参个数实例
2014/08/27 Javascript
深入解析JavaScript编程中的this关键字使用
2015/11/09 Javascript
jquery封装插件时匿名函数形参和实参的写法解释
2017/02/14 Javascript
seajs模块压缩问题与解决方法实例分析
2017/10/10 Javascript
Javascript 关于基本类型和引用类型的个人理解
2019/11/01 Javascript
在 Vue 中编写 SVG 图标组件的方法
2020/02/24 Javascript
Python面向对象编程中的类和对象学习教程
2015/03/30 Python
tensorflow 用矩阵运算替换for循环 用tf.tile而不写for的方法
2018/07/27 Python
python读取目录下最新的文件夹方法
2018/12/24 Python
pandas 把数据写入txt文件每行固定写入一定数量的值方法
2018/12/28 Python
python datetime中strptime用法详解
2019/08/29 Python
python程序 创建多线程过程详解
2019/09/23 Python
pytorch中的weight-initilzation用法
2020/06/24 Python
python中id函数运行方式
2020/07/03 Python
Python实现敏感词过滤的4种方法
2020/09/12 Python
html5使用html2canvas实现浏览器截图的示例
2017/08/31 HTML / CSS
竞聘副主任科员演讲稿
2014/01/11 职场文书
环保倡议书400字
2014/05/15 职场文书
股东合作协议书
2014/09/12 职场文书
简单租房协议书(范本)
2014/10/13 职场文书
学校法制宣传日活动总结
2014/11/01 职场文书
百年校庆感言
2015/08/01 职场文书
公司员工管理制度
2015/08/04 职场文书
python保存大型 .mat 数据文件报错超出 IO 限制的操作
2021/05/10 Python
SQL注入的实现以及防范示例详解
2021/06/02 MySQL
WCG2010 星际争霸决赛 Flash vs Goojila 1 星际经典比赛回顾
2022/04/01 星际争霸
JS前端可视化canvas动画原理及其推导实现
2022/08/05 Javascript