JS 预编译代码实例分析

了解 JavaScript 引擎在执行代码过程中所做的一些行为是非常必要的,这有助于我们在遇到莫名其妙的调用时,能够大致定位问题所在。在我学习了预编译的相关知识,并基于该文章,引用其中的一段代码,结合"变量提升"、"函数提升"的小示例,对其进行详细的分析,算是留作一份笔记巩固记忆、加深理解。

代码

javascript 复制代码
console.log(a)
fn1(1)
var a = 123
console.log(a)

var fn1 = () => {
  console.log(a)
}

function fn1(a) {
  console.log(a)
  var a = 666
  console.log(a)
  function a() {}
  console.log(a)
  var b = function () {}
  console.log(b)
  function c() {}
}

fn1(1)

错误的推导会让你认为上述代码的打印如下:

javascript 复制代码
如果你判断首行报错,那么需要了解变量提升
或者你这样认为
undefined
undefined
666
[Function: a]
[Function: b]
123
undefined
666
[Function: a]
[Function: b]

实际上,上方的代码打印如下:

javascript 复制代码
undefined
[Function: a]
666
666
[Function: b]
123
123

详细分析

1. 创建全局对象 GO

在全局执行上下文中,创建全局对象 GO

2. 加载当前 JS 文件

加载并解析当前的 JavaScript 文件。

3. 脚本语法分析

进行语法分析,确保代码没有语法错误。

4. 当前 JS 文件预编译

4-1. 查找变量声明
javascript 复制代码
GO = {
  a: undefined
}
4-2. 查找函数声明(除了函数表达式)
javascript 复制代码
GO = {
  a: undefined,
  fn1: function fn1(a) {}
}

5. 正常执行(执行到函数调用前)

javascript 复制代码
console.log(a) // 打印 undefined
fn1(1) // 执行到这里了,小心,函数也有预编译,执行前一刻完成

6. 函数预编译

6-1. 创建活跃对象 AO
javascript 复制代码
AO = {}
6-2. 查找变量和形参
javascript 复制代码
AO = {
  a: undefined,
  b: undefined
}
6-3. 实参值和形参统一
javascript 复制代码
AO = {
  a: 1,
  b: undefined
}
6-4. 查找函数(非函数表达式)
javascript 复制代码
AO = {
  a: function a() {},
  b: undefined,
  c: function c() {}
}

7. 正常执行函数(根据 AO)

javascript 复制代码
console.log(a)  // 打印 function a() {}
var a = 666  // a 改变,AO.a = 666
console.log(a)  // 打印 666
function a() {}  // 该声明已提升过,不会覆盖
console.log(a)  // 打印 666
var b = function () {}  // b 改变,AO.b = function () {}
console.log(b)  // 打印 function () {}
function c() {}  // 该声明已提升过,不会覆盖

8. 接着执行函数外代码,执行到下个函数调用前

javascript 复制代码
fn1(1) // 已讲述,上续
var a = 123  // GO 对象中的 a 改变为 123(undefined > 123)
console.log(a)  // 打印 123

var fn1 = () => {  // fn1 改变,GO.fn1 = () => {...}
  console.log(a)
}

function fn1(a) {  // 该声明已提升过(函数提升),不会覆盖
  ...
}

fn1(1)  // 执行到这里时,预编译

9. 函数预编译

9-1. 创建活跃对象 AO
javascript 复制代码
AO = {}
9-2. 查找变量和形参
javascript 复制代码
AO = {
  a: undefined
}
9-3. 实参值和形参统一
javascript 复制代码
AO = {
  a: 1
}
9-4. 查找函数(非函数表达式)
javascript 复制代码
AO = {
  a: 1
}

10. 正常执行函数(根据 AO)

javascript 复制代码
console.log(a)  // a 不存在当前函数作用域,往上级查找,找到 GO.a,打印 123

总结

  • 全局预编译:创建 GO 对象,查找变量声明和函数声明。
  • 函数预编译:创建 AO 对象,查找变量和形参,实参值和形参统一,查找函数声明。
  • 执行阶段:按照代码顺序执行,变量赋值和函数调用。
相关推荐
柒和远方3 小时前
Phase 7.4 学习博客:为什么多 API 项目需要 Swagger / OpenAPI
前端·后端·架构
张龙6873 小时前
拼多多开放平台对接踩坑实录:从 CLIENT_ID 配置到 MD5 签名算法的完整填坑指南
前端
GuWenyue3 小时前
提示词彻底过时?一套上下文工程方案,3步让LLM落地生产,代码直接复用
前端·javascript·人工智能
柒和远方3 小时前
Phase 7.3 复盘:后台任务不只是“扔进队列”,还要能被看见
前端·后端·架构
2501_943782353 小时前
【共创季稿事节】 倒计时器:时分秒选择器与定时器的协同工作
前端·华为·harmonyos·鸿蒙·鸿蒙系统
奶油mm3 小时前
公司技术债堆积如山,我一人之力用 Vue3 偷换了整个前端架构
前端·vue.js
用户938515635073 小时前
深入理解 JavaScript 中的 this 与数据存储的奥秘
前端·javascript
JNX_SEMI4 小时前
AT2659 L1频段多模卫星导航低噪声放大器技术解析
前端·单片机·嵌入式硬件·物联网·硬件工程
Profile排查笔记5 小时前
指纹浏览器环境异常排查:Fingerprint、Profile、Proxy、Session 和 Task Log 怎么看
前端·人工智能·后端·自动化
京韵养生记6 小时前
【无标题】
java·服务器·前端