JavaScript运行原理分析


Posted in Javascript onFebruary 09, 2018

JavaScript是一种基于对象的动态、弱类型脚本语言(以下简称JS),是一种解释型语言,和其他的编程语言不同,如java/C++等编译型语言,这些语言在代码执行前会进行通篇编译,先编译成字节码(机器码)。然后在执行。而JS不是这样做的,JS是不需要编译成中间码,而是可以直接在浏览器中运行,JS运行过程可分为两个阶段,编译和执行。(可参考你不知道的JS这本书),当JS控制器转到一段可执行的代码时(这段可执行代码就是编译阶段生成的),会创建与之对应的执行上下文(Excution Context简称EC)。执行上下文可以理解为执行环境(执行上下文只能由JS解释器创建,也只能由JS解释器使用,用户是不可以操作该‘对象'的)。

JS中的执行环境分为三类:

  • 全局环境:当JS引擎进入一个代码块时,如遇到<script>xxx</script>标签,就是进入一个全局执行环境
  • 函数环境:当一个函数被调用时,在函数内部就形成了一个函数执行环境
  • eval():把字符串单做JS代码执行,不推荐使用

在一段JS代码中可能会产生多个执行上下文,在JS中用栈这种数据结构来管理执行上下文,栈的特点是“先进后出,后进先出”,这种栈称之为函数调用栈。

执行上下文的特点

  • 栈底永远是全局执行上下文,有且仅有一个
  • 全局执行上下文只有在浏览器关闭时,才会弹出栈
  • 其他的执行上下文的数量没有限制
  • 栈顶永远是当前活动执行上下文,其余的都处于等待状态中,一旦执行完毕,立即弹出栈,然后控制权交回下一个执行上下文
  • 函数只有在每次被调用时,才会为其创建执行上下文,函数被声明时是没有的。

执行上下文可以形象的理解为一个普通的JS对象,一个执行上下文的生命周期大概包含两个阶段:

创建阶段

此阶段主要完成三件事件,1、创建变量对象 2、建立作用域链 3、确定this指向

执行阶段

此阶段主要完成变量赋值、函数调用、其他操作

变量对象(VO)的创建过程

  • 1、根据函数参数,创建并初始化arguments对象,给arguments对象添加属性"0","1","2","3"等属性,其初始值为undefined,并设置arguments.length值为实际传入参数的个数。
  • 2、查找function函数声明,在变量对象上添加属性,属性名就是函数名,属性值就是函数的引用值,如果已经存在同名的,则直接覆盖
  • 3、查找var变量声明(查找变量时,会把函数的参数等价于var声明,所以在VO中也会添加和参数名一样的属性,初始值也是undefined),在变量对象添加属性,属性名就是变量名,属性值是undefined,如果已经存在同名的,则不处理

如果存在同名标识符(函数、变量),则函数可以覆盖变量,函数的优先级高于变量

变量对象(OV)和激活对象(AO)是同一个东西,在不同时期的两种叫法。在创建时期叫变量对象,在执行时期叫激活对象

以如下代码为例

var g_name="tom";
var g_age=20;
function g_fn(num){
 var l_name="kity";
 var l_age=18;
 function l_fn(){
  console.log(g_name + '===' + l_name + '===' + num);
 }
}
g_fn(10);

编译阶段

当JS控制器转到这一段代码时,会创建一个执行上下文,G_EC

执行上下文的结构大概如下:

G_EC = {
 VO   : {},
 Scope_chain : [],
 this  : {}
}

/* VO的结构大概 */
VO = {
 g_name : undefined,
 g_age : undefined,
 g_fn : <函数在内存中引用值>
}

/* Scope_chain的大概结构如下 */
Scope_chain = [ G_EC.VO ] // 数组中第一个元素是当前执行上下文的VO,第二个是父执行上下文的VO,最后一个是全局执行上下文的VO,在执行阶段,会沿着这个作用域链一个一个的查找标识符,如果查到则返回,否知一直查找到全局执行上下文的VO

/* this */
this = undefined // 此时this的值是undefined

执行上下文一旦创建完毕,就立马被压入函数调用栈中,此时解释器会悄悄的做一件事情,就是给当前VO中的函数添加一个内部属性[[scope]],该属性指向上面的作用域链。

g_fn.scope = [ global_EC.VO ] // 该scope属性只能被JS解释器所使用,用户无法使用

执行阶段

一行一行执行代码,当遇到一个表达式时,就会去当前作用域链的中查找VO对象,如果找到则返回,如果找不到,则继续查找下一个VO对象,直至全局VO对象终止。

此阶段可以有变量赋值,函数调用等操作,当解释器遇到g_fn()时,就知道这是一个函数调用,然后立即为其创建一个函数执行上下文,fn_EC,该上下文fn_EC同样有两个阶段

分别是创建阶段和执行阶段。

在创建阶段,对于函数执行上下文,在创建变量对象时,会多创建一个arguments对象,然后为arguments对象添加属性:"0","1", "2"其初始值为undefined,

  • 查找function函数声明
  • 查找var变量声明
Javascript 相关文章推荐
javascript克隆对象深度介绍
Nov 20 Javascript
js动态添加事件并可传参数示例代码
Oct 21 Javascript
jQuery实现在下拉列表选择时获取json数据的方法
Apr 16 Javascript
在线所见即所得HTML编辑器的实现原理浅析
Apr 25 Javascript
javascript实现图片延迟加载方法汇总(三种方法)
Aug 27 Javascript
基于javascript实现tab选项卡切换特效调试笔记
Mar 30 Javascript
JS/jQuery判断DOM节点是否存在的简单方法
Nov 24 Javascript
vue mint-ui 实现省市区街道4级联动示例(仿淘宝京东收货地址4级联动)
Oct 16 Javascript
《javascript少儿编程》location术语总结
May 27 Javascript
详解在Javascript中进行面向切面编程
Apr 28 Javascript
vue iview多张图片大图预览、缩放翻转
Jul 13 Javascript
JS数组Reduce方法功能与用法实例详解
Apr 29 Javascript
vue 全选与反选的实现方法(无Bug 新手看过来)
Feb 09 #Javascript
详解如何在项目中使用jest测试react native组件
Feb 09 #Javascript
vue checkbox 全选 数据的绑定及获取和计算方法
Feb 09 #Javascript
mint-ui 时间插件使用及获取选择值的方法
Feb 09 #Javascript
VUE2 前端实现 静态二级省市联动选择select的示例
Feb 09 #Javascript
尝试自己动手用react来写一个分页组件(小结)
Feb 09 #Javascript
在 React、Vue项目中使用SVG的方法
Feb 09 #Javascript
You might like
PHP在XP下IIS和Apache2服务器上的安装
2006/09/05 PHP
Php连接及读取和写入mysql数据库的常用代码
2014/08/11 PHP
PHP框架Swoole定时器Timer特性分析
2014/08/19 PHP
Yii的CDbCriteria查询条件用法实例
2014/12/04 PHP
php 使用redis锁限制并发访问类示例
2016/11/02 PHP
JS控制阿拉伯数字转为中文大写示例代码
2013/09/04 Javascript
使用firebug进行调试javascript的示例
2013/12/16 Javascript
JavaScript中的数学运算介绍
2014/12/29 Javascript
setTimeout内不支持jquery的选择器的解决方案
2015/04/28 Javascript
简述JavaScript的正则表达式中test()方法的使用
2015/06/16 Javascript
jQuery超酷平面式时钟效果代码分享
2020/03/30 Javascript
javascript设置页面背景色及背景图片的方法
2015/12/29 Javascript
bootstrap输入框组代码分享
2016/06/07 Javascript
原生js实现tab选项卡切换
2020/03/23 Javascript
BootStrap树状图显示功能
2016/11/24 Javascript
微信小程序使用Promise简化回调
2018/02/06 Javascript
微信小程序自定义弹窗wcPop插件
2018/11/19 Javascript
微信小程序实现批量倒计时功能
2020/11/01 Javascript
vue 路由懒加载中给 Webpack Chunks 命名的方法
2020/04/24 Javascript
[01:15:36]加油刀塔第二期网络版
2014/08/09 DOTA
[51:36]EG vs VP 2018国际邀请赛淘汰赛BO3 第一场 8.24
2018/08/25 DOTA
python正则表达式re模块详细介绍
2014/05/29 Python
Linux下用Python脚本监控目录变化代码分享
2015/05/21 Python
Python实现包含min函数的栈
2016/04/29 Python
详解Python 序列化Serialize 和 反序列化Deserialize
2017/08/20 Python
Python 多进程并发操作中进程池Pool的实例
2017/11/01 Python
Python中注释(多行注释和单行注释)的用法实例
2019/08/28 Python
使用pyshp包进行shapefile文件修改的例子
2019/12/06 Python
第一范式(1NF)、第二范式(2NF)和第三范式(3NF)之间的区别是什么?
2016/04/28 面试题
高级技校毕业生自荐信
2013/11/18 职场文书
怎样写好创业计划书的内容
2014/02/06 职场文书
元宵节晚会主持人串词
2014/03/25 职场文书
《爱如茉莉》教后反思
2014/04/12 职场文书
毕业论文答辩开场白和结束语
2015/05/27 职场文书
2016年重阳节慰问信
2015/12/01 职场文书
Python 居然可以在 Excel 中画画你知道吗
2022/02/15 Python