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

相关推荐
Mintopia22 分钟前
实时 AIGC:Web 端低延迟生成的技术难点与突破
前端·javascript·aigc
Mintopia24 分钟前
Next.js 性能优化双绝:Image 与 next/font 的底层修炼手册
前端·javascript·next.js
小菜全30 分钟前
使用UniApp实现下拉框和表格组件页面
开发语言·前端·javascript·elementui·前端框架·html
傻梦兽2 小时前
2025年,跟 encodeURIComponent() 说再见吧
前端·javascript
Lingxing2 小时前
事件流:深入理解事件冒泡、事件捕获与事件委托
前端·javascript·面试
San302 小时前
JavaScript 入门精要:从变量到对象,构建稳固基础
javascript·面试·html
bug_kada2 小时前
深入理解事件捕获与冒泡(详细版)
前端·javascript
小妖6662 小时前
怎么用 tauri 创建一个桌面应用程序(Electron)
前端·javascript·electron
EndingCoder2 小时前
单元测试:Jest 与 Electron 的结合
javascript·electron·单元测试·前端框架