JavaScript从入门到入土(3):预编译

预编译(通常称为 "变量提升")是 JavaScript 的一种特殊机制,发生在编译阶段的早期 。它会将变量和函数声明 "提升" 到当前作用域的顶部,但赋值不会提升

编译过程

v8 引擎读取到js代码后,先编译后执行

编译全局 ->执行全局(有函数) ->编译函数 ->执行函数 ->继续执行全局

函数体编译过程

  1. 创建函数的上下文执行对象
  2. 找形参和变量(包括函数表达式)声明,将形参和变量名作为对象的属性名,值为undefined
  3. 将实参和形参相统一
  4. 在函数体里面找函数声明(不管函数表达式),函数名作为上下文对象的属性名,值为函数体

全局编译过程

  1. 创建全局执行上下文对象
  2. 找变量声明,将变量名作为对象的属性名,值为undefined
  3. 在全局找函数声明,函数名作为上下文对象的属性名,值为函数体

练习

代码最后输出什么?

js 复制代码
function foo(a, b) {
  console.log(a);
  c = 0
  var c;
  a = 3
  b = 2
  console.log(b);
  function b() {}
  console.log(b);
}
foo(1)

编译全局

  1. 创建全局执行上下文对象
  2. 未找到变量声明
  3. 找到函数声明,建立对象属性foo,值为函数体

最后得到如下的全局执行上下文对象

js 复制代码
GLOBAL:{
    foo: 函数体
}

执行全局

执行foo(1),编译foo函数

编译函数

  1. 创建函数的上下文执行对象
  2. 找到形参a、b和变量变量声明c作为对象属性名,值为undefined
  3. 将实参和形参相统一,把实参1赋值给形参a
  4. 在函数体里面找到函数声明function b() ,建立对象属性b,值为函数体(因为已经存在属性b,所以将b的undefined属性改为函数体)。

最后得到如下函数的上下文执行对象

js 复制代码
FUNCTION:{
  a:/*undefined  ->*/ 1 
  b:/*undefined ->*/函数体
  c:undefined
}

执行函数

  1. 执行打印语句console.log(a);,在函数的上下文执行对象中找到属性a的值为1,则输出 1
  2. 执行赋值语句c = 0,将属性c的值改为0
  3. 执行赋值语句a = 3,将属性a的值改为3
  4. 执行赋值语句b = 2,将属性b的值改为2
  5. 执行打印语句console.log(b);,在函数的上下文执行对象中找到属性b的值为2,则输出 2
  6. 执行打印语句console.log(b);,在函数的上下文执行对象中找到属性b的值为2,则输出 2

因为全局只需执行一个函数,所以当函数执行完毕全局也执行完毕,至此整个程序的编译执行过程我们已经分析完成。

答案为 :

1

2

2

相关推荐
paopaokaka_luck2 小时前
基于SpringBoot+Uniapp的健身饮食小程序(协同过滤算法、地图组件)
前端·javascript·vue.js·spring boot·后端·小程序·uni-app
Ares-Wang5 小时前
JavaScript》》JS》 Var、Let、Const 大总结
开发语言·前端·javascript
SY_FC6 小时前
uniapp input 聚焦时键盘弹起滚动到对应的部分
javascript·vue.js·elementui
渣渣盟8 小时前
JavaScript核心概念全解析
开发语言·javascript·es6
Carlos_sam9 小时前
OpenLayers:ol-wind之渲染风场图全解析
前端·javascript
拾光拾趣录9 小时前
闭包:从“变量怎么还没死”到写出真正健壮的模块
前端·javascript
拾光拾趣录9 小时前
for..in 和 Object.keys 的区别:从“遍历对象属性的坑”说起
前端·javascript
遂心_10 小时前
深入解析前后端分离中的 /api 设计:从路由到代理的完整指南
前端·javascript·api
@大迁世界11 小时前
第7章 React性能优化核心
前端·javascript·react.js·性能优化·前端框架
DownToEarth11 小时前
H5实现获取当前定位
javascript