详细了解JavaScript编译原理
作者:互联网
2022-07-22

1.分词/词法分析
首先来说一下什么是分词吧,分词就是将由字符组成的字符串分解成对于编程语言有意义的代码块。这些代码块就被成为词法单元。如var a = 2这段代码,会被分解为下面的词法单元。具体为:var,a,=,2。
注意:分词其实就是将上面的整段代码进行一个拆分为一段一段。
2.解析/语法分析
解析就是将词法单元流转换为一个有元素逐级嵌套所组成的代表了程序语法结构的树。这个树被称为:抽象语法树。鉴于这里过长的标准词,就不做考虑了,本人直接以一种更加直观的形式展示。具体如下图:
解析:抽象语法树会有一个var的顶级节点,之后会有一个变量为a的子节点以及赋值符=的一个节点。在赋值符下又有一个为2的子节点。具体就对应了var a = 2这段代码。
3.代码生成
将抽象语法树转换为可执行代码的过程被称为代码生成。这个过程与语言,目标平台息息相关。简单来说就是有某种方法可以将var a = 2的抽象语法树转换为机器的指令。用来创建一个叫做a的变量,并将一个值存储在a中。
4.LHS和RHS查询
执行JavaScript代码主要是依赖引擎。当引擎执行var a = 2时,会通过查找变量a来判断是否已经声明。查找的过程由作用域协助。在查询的过程中,引擎会为变量a进行LHS(左查询),会为值进行右查询。简单来说就是当变量出现在赋值操作的左侧时,进行LHS查询,出现在右侧时,进行RHS查询。更准确来说就是LHS查询试图找到变量的容器本身,RHS则是为了取到他的源值。
注意:在函数中,会出现既有LHS又有RHS查询。因为在传递参数的过程中,会代码会进行隐式的赋值。
5.异常
当变量还没声明的情况下,LHS查询和RHS查询的行为是不一样的。
function foo(a){
console.log(a+b);
b=a;}foo(2)注意:第一次对b进行右查询是无法找到该变量的,也就是说这是一个未声明的变量,因为在任何相关的作用域中都无法找到他。如果RHS在所嵌套的作用域中遍寻不到所需的变量,引擎就会抛出异常。
6.小测验
function foo(a){
var b=a;
return a+b;
}
var c=foo(2)问题:找出所有的LHS查询和RHS
答案:LHS(c=…,a=2,b=…)和RHS(foo(2…,=a,a…,…b))
相关标签:
相关推荐
专题
+ 收藏
+ 收藏
+ 收藏
+ 收藏
+ 收藏
+ 收藏
最新数据
相关文章
Vue3 KeepAlive 深度揭秘:组件缓存的魔法是如何实现的?
你的 Vue 3 useAttrs(),VuReact 会编译成什么样的 React?
虚拟 DOM 的 Diff 算法:Vue/React 如何实现高效更新
前端必看!console 调试不只有 log,这 8 个技巧省一半调试时间
在线CAD开发包图纸转换功能使用指南
Vue组件通信全场景详解(Vue2+Vue3适配)| 实战必备,新手也能看懂
vite+vue2 动态路由加载方法实现
「性能优化」虚拟列表极致优化实战:从原理到源码,打造丝滑滚动体验
你的 Vue 3 生命周期,VuReact 会编译成什么样的 React?
你的 Vue 3 defineProps(),VuReact 会编译成什么样的 React?
AI精选
MCP协议设计与实现-第20章 从零构建一个生产级 MCP Server
MCP协议设计与实现-第16章 服务发现与客户端注册
MCP协议设计与实现-第18章 Elicitation、Roots 与配置管理
MCP协议设计与实现-第10章 Python Server 实现剖析
MCP协议设计与实现-第17章 sampling
MCP协议设计与实现-第09章 TypeScript Client 实现剖析
MCP协议设计与实现-第19章 Claude Code 的 MCP 客户端:12 万行的实战
MCP协议设计与实现-第12章 STDIO 传输:本地进程通信
