了解 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)

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

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

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

undefined
[Function: a]
666
666
[Function: b]
123
123

详细分析

1. 创建全局对象 GO

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

2. 加载当前 JS 文件

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

3. 脚本语法分析

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

4. 当前 JS 文件预编译

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

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

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

6. 函数预编译

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

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

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. 接着执行函数外代码,执行到下个函数调用前

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
AO = {}
9-2. 查找变量和形参
AO = {
a: undefined
}
9-3. 实参值和形参统一
AO = {
a: 1
}
9-4. 查找函数(非函数表达式)
AO = {
a: 1
}

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

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

总结

  • 全局预编译:创建 GO 对象,查找变量声明和函数声明。
  • 函数预编译:创建 AO 对象,查找变量和形参,实参值和形参统一,查找函数声明。
  • 执行阶段:按照代码顺序执行,变量赋值和函数调用。

JS 预编译代码实例分析的更多相关文章

  1. js预编译

    先来做三个测试 eg1: var a; a = 1; function a() {}; console.log(a); eg2: var a; function a() {}; console.log ...

  2. Handlebars.js 预编译(转)

    Handlebars.js 官网上对预编译1是这样说的: 你需要安装 Node.js 你需要在全局环境中,通过 Npm 安装 handlebars 包 然后你就可以通过命令预编译你的 handleba ...

  3. 一个proc预编译代码时coredump的问题分析

    最近有同事在搞编译环境迁移,碰上一个问题让我帮他看一下.    他建了一个新目录,然后把现在的代码拷过去,编译的时候发现有一个文件编译不了一执行就出现core,不知道啥情况.    我进到他的编译环境 ...

  4. js 预编译

    js 运行代码的时候分为几个步骤:语法分析 ==>预编译  ==>解释执行 语法解析:通篇扫描代码,查看语法是否出错 解释执行:读一行 - 解释一行 - 执行一行 预编译执行的操作: // ...

  5. 关于js预编译以及js文件执行顺序的几个问题。

    关于js的执行原理,除去html页面中直接添加的代码,js代码的放置可以分为两类. //情形a           <script type="text/javascript" ...

  6. JS预编译详解

    我们都知道javascript是解释型语言,执行的特点呢是编译一行,执行一行.按照这个思路有时候我们在运行代码时会有一些令人费解的现象出现.下面我们一起来执行下面三段代码. <script> ...

  7. js预编译和函数执行

    javascript 执行过程 1.语法检测(有没有基本的语法错误,例如中文,关键字错误...)2.词法分析(预编译) (1)创建全局GO(global object)对象 (2)对var声明的变量进 ...

  8. js预编译的四部曲

    众所周知javascript是解释性语言,主要特点为解释一行执行一行. 而在js运行时会进行三件事:1语法分析  2.预编译  3.解释执行 语法分析会在代码执行前对代码进行通篇检查,以排除一些低级错 ...

  9. JS预编译过程

    GO和AO 变量的预编译 实例1 console.log(a); var a=1; console.log(a); 实际编译过程: 将a存入预编译对象中,赋值为undefined: 真正的赋值语句当程 ...

  10. js预编译环节 变量声明提升 函数声明整体提升

    预编译四部曲 1.创建AO对象 2.找形参和变量声明,将变量和形参名作为AO属性名,值为undefined 3.将实参和形参统一 4.在函数体里面找函数声明,值赋予函数体 function fn(a) ...

随机推荐

  1. 以太坊Rollup方案之 arbitrum(1)

    什么是Rollup? 以太坊的Rollup扩容是一种Layer 2(第二层)扩容解决方案,旨在提高以太坊区块链的交易吞吐量和性能.它通过将大量的交易数据转移到以太坊区块链之外的第二层网络来实现这一目标 ...

  2. 全网最适合入门的面向对象编程教程:49 Python函数方法与接口-函数与方法的区别和lamda匿名函数

    全网最适合入门的面向对象编程教程:49 Python 函数方法与接口-函数与方法的区别和 lamda 匿名函数 摘要: 在 Python 中,函数和方法都是代码的基本单元,用于封装和执行特定的任务:它 ...

  3. GPUStack正式发布: 为大模型而生的开源GPU集群管理器

    经过 Seal 研发团队几个月来持续的开发和测试,我们满怀期待及热情地发布新的产品 GPUStack,GPUStack 是一个用于运行 LLM(大型语言模型)的开源 GPU 集群管理器.尽管如今大语言 ...

  4. WebGL学习笔记

    完整 demo 和 lib 文件可以在 https://github.com/tengge1/webgl-guide-code 中找到. 第 2 章 WebGL 入门 第一个 WebGL 程序 fun ...

  5. 《Vue.js 设计与实现》读书笔记 - 第13章、异步组件与函数式组件

    第13章.异步组件与函数式组件 13.1 异步组件要解决的问题 用户可以简单通过 import 异步导入组件. <template> <component :is="asy ...

  6. Linux内核中cpu_capacity是什么?

    cpu_capacity 在Linux内核中,cpu_capacity 是用于表示每个CPU的处理能力的一个参数,通常用于调度器的负载均衡.它表明不同的CPU核心在计算资源分配中的相对性能,尤其在异构 ...

  7. Linux内存管理2.6 -反向映射RMAP(最终版本)

    所谓反向映射是相对于从虚拟地址到物理地址的映射,反向映射是从物理页面到虚拟地址空间VMA的反向映射. RMAP能否实现的基础是通过struct anon_vma.struct anon_vma_cha ...

  8. 封装一个Promise.all 的函数

    // 1. 准备三个异步函数 const promise1 = Promise.resolve('prom11ise1'); const promise2 = new Promise(function ...

  9. 在实例化对象的时候new关键字具体做了哪些操作?

    a 创建了一个空对象 {}b 通过原型链把空对象和构造函数连接起来__proto__ = prototype c 构造函数的this指向新对象,并执行函数体 d 判断构造函数的返回值,返回对象就使用该 ...

  10. 在 KubeSphere 上快速安装和使用 KDP 云原生数据平台

    作者简介:金津,智领云高级研发经理,华中科技大学计算机系硕士.加入智领云 8 余年,长期从事云原生.容器化编排领域研发工作,主导了智领云自研的 BDOS 应用云平台.云原生大数据平台 KDP 等产品的 ...