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

相关推荐
烛阴1 小时前
比UUID更快更小更强大!NanoID唯一ID生成神器全解析
前端·javascript·后端
Alice_hhu1 小时前
ResizeObserver 解决 echarts渲染不出来,内容宽度为 0的问题
前端·javascript·echarts
charlee442 小时前
解决Vditor加载Markdown网页很慢的问题(Vite+JS+Vditor)
javascript·markdown·cdn·vditor
技术小丁4 小时前
使用 HTML + JavaScript 在高德地图上实现物流轨迹跟踪系统
前端·javascript·html
stark张宇4 小时前
Web - 面向对象
前端·javascript
代码的余温4 小时前
打造极致计算器:HTML+Tailwind+DaisyUI实战
javascript·css·html
waterHBO4 小时前
vscode 插件 eslint, 检查 js 语法
javascript·ide·vscode
sunbyte5 小时前
50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Form Wave(表单label波动效果)
前端·javascript·css·vue.js·tailwindcss
拉不动的猪5 小时前
面试题之JS中,有哪些方法可以终止for循环
前端·javascript·面试
爱看书的小沐6 小时前
【小沐杂货铺】基于Three.JS构建IFC模型浏览器(WebGL、CAD、Revit、IFC)
javascript·webgl·three.js·bim·ifc·revit·ifc.js