找回密码
 会员注册
查看: 28|回复: 0

Js是怎样运行起来的?_UTF_8

[复制链接]

2万

主题

0

回帖

7万

积分

超级版主

积分
75285
发表于 2024-9-30 00:06:29 | 显示全部楼层 |阅读模式
前言不知道大家有没有想过这样一个问题,我们所写的 JavaScript 代码是怎样被计算机认识并且执行的呢?这中间的过程具体是怎样的呢?有的同学可能已经知道,Js 是通过 Js 引擎运行起来的,那么什么是 Js 引擎?Js 引擎是怎样编译执行和优化 Js 代码的?Js 引擎有很多种,比如 Chrome 使用的 V8 引擎,Webkit 使用的是 JavaScriptCore,React Native 使用的是 Hermes。今天我们主要来分析一下比较主流的 V8 引擎是怎样运行 Js 的。V8 引擎在介绍 V8 引擎的概念之前,我们先来回顾一下编程语言。编程语言可以分为机器语言、汇编语言、高级语言。机器语言:由 0 和 1 组成的二进制码,对于人类来说是很难记忆的,还要考虑不同 CPU 平台的兼容性。汇编语言:用更容易记忆的英文缩写标识符代替二进制指令,但还是需要开发人员有足够的硬件知识。高级语言:更简单抽象且不需要考虑硬件,但是需要更复杂、耗时更久的翻译过程才能被执行。到了这里我们知道,高级语言一定要转化为机器语言才能被计算机执行,而且越高级的语言转化的时间越久。高级语言又可以分为解释型语言、编译型语言。编译型语言:需要编译器进行一次编译,被编译过的文件可以多次执行。如 C++、C 语言。解释型语言:不需要事先编译,通过解释器一边解释一边执行。启动快,但执行慢。我们知道 JavaScript 是一门高级语言,并且是动态类型语言,我们在定义一个变量时不需要关心它的类型,并且可以随意的修改变量的类型。而在像 C++这样的静态类型语言中,我们必须提前声明变量的类型并且赋予正确的值才行。也正是因为 JavaScript 没有像 C++那样可以事先提供足够的信息供编译器编译出更加低级的机器代码,它只能在运行阶段收集类型信息,然后根据这些信息进行编译再执行,所以 JavaScript 也是解释型语言。这也就意味着 JavaScript 要想被计算机执行,需要一个能够快速解析并且执行 JavaScript 脚本的程序,这个程序就是我们平时所说的 JavaScript 引擎。这里我们给出 V8 引擎的概念:V8 是 Google 基于 C++ 编写的开源高性能 Javascript 与 WebAssembly 引擎。用于 Google Chrome(Google 的开源浏览器) 以及 Node.js 等。CPU 是如何执行机器指令的?将高级语言转化为机器语言之后,CPU 又是怎样执行的呢?我们以一段 C 代码为例:intmain(){intx=1;inty=2;intz=x+y;returnz;}}先来看一下以上代码被转换为机器语言是什么样子。下图左侧是用十六进制表示的二进制机器码,中间部分是汇编代码,右侧是指令的含义。CPU 执行机器指令的流程首先程序在执行之前会被装进内存。系统会将二进制代码中的第一条指令的地址写入到 PC 寄存器中。CPU 根据 PC 寄存器中的地址,从内存中取出指令。将下一条指令的地址更新到 PC 寄存器中。分析当前取出指令,并识别出不同的类型的指令,以及各种获取操作数的方法。加载指令:从内存中复制指定长度的内容到通用寄存器中,并覆盖寄存器中原来的内容。存储指令:将寄存器中的内容复制到内存某个位置,并覆盖掉内存中的这个位置上原来的内容。上图中 movl 指令后面的 %ecx 就是寄存器地址,-8(%rbp) 是内存中的地址,这条指令的作用是将寄存器中的值拷贝到内存中。更新指令:复制两个寄存器中的内容到 ALU 中,也可以是一块寄存器和一块内存中的内容到 ALU 中,ALU 将两个字相加,并将结果存放在其中的一个寄存器中,并覆盖该寄存器中的内容。...执行指令完毕,进入下一个 CPU 时钟周期。V8 引擎的编译流水线接下来我们先从宏观的角度来看一下 V8 是怎么执行 JavaScript 代码的,然后再对每一步进行分析。初始化基础环境;解析源码生成 AST 和作用域;依据 AST 和作用域生成字节码;解释执行字节码;监听热点代码;...完整的分析一段 JavaScript 代码是怎样被执行的1、初始化基础环境V8 执行 Js 代码是离不开宿主环境的,V8 的宿主可以是浏览器,也可以是 Node.js。下图是浏览器的组成结构,其中渲染引擎就是平时所说的浏览器内核,它包括网络模块,Js 解释器等。当打开一个渲染进程时,就为 V8 初始化了一个运行时环境。运行时环境为 V8 提供了堆空间,栈空间、全局执行上下文、消息循环系统、宿主对象及宿主 API 等。V8 的核心是实现了 ECMAScript 标准,此外还提供了垃圾回收器等内容。2、解析源码生成 AST 和作用域基础环境准备好之后,接下来就可以向 V8 提交要执行的 JavaScript 代码了。首先 V8 会接收到要执行的 JavaScript 源代码,不过这对 V8 来说只是一堆字符串,V8 并不能直接理解这段字符串的含义,它需要结构化这段字符串。functionadd(x,y){varz=x+yreturnz}console.log(add(1,2))比如针对如上源代码,V8 首先通过解析器(parser)解析成如下的抽象语法树 AST:[generatingbytecodeforfunction:add]---AST---FUNCat12.KIND0.LITERALID1.SUSPENDCOUNT0.NAME"add".PARAMS..VAR(0x7fa7bf8048e8)(mode=VAR,assigned=false)"x"..VAR(0x7fa7bf804990)(mode=VAR,assigned=false)"y".DECLS..VARIABLE(0x7fa7bf8048e8)(mode=VAR,assigned=false)"x"..VARIABLE(0x7fa7bf804990)(mode=VAR,assigned=false)"y"..VARIABLE(0x7fa7bf804a38)(mode=VAR,assigned=false)"z".BLOCKNOCOMPLETIONSat-1..EXPRESSIONSTATEMENTat31...INITat31....VARPROXYlocal[0](0x7fa7bf804a38)(mode=VAR,assigned=false)"z"....ADDat32.....VARPROXYparameter[0](0x7fa7bf8048e8)(mode=VAR,assigned=false)"x".....VARPROXYparameter[1](0x7fa7bf804990)(mode=VAR,assigned=false)"y".RETURNat37..VARPROXYlocal[0](0x7fa7bf804a38)(mode=VAR,assigned=false)"z"V8 在生成 AST 的同时,还生成了 add 函数的作用域:Globalscope:functionadd(x,y){//(0x7f9ed7849468)(12,47)//willbecompiled//1stackslots//localvars:VARy;//(0x7f9ed7849790)parameter[1],neverassignedVARz;//(0x7f9ed7849838)local[0],neverassignedVARx;//(0x7f9ed78496e8)parameter[0],neverassigned}在解析期间,所有函数体中声明的变量和函数参数,都被放进作用域中,如果是普通变量,那么默认值是 undefined,如果是函数声明,那么将指向实际的函数对象。在执行阶段,作用域中的变量会指向堆和栈中相应的数据。3、依据 AST 和作用域生成字节码生成了作用域和 AST 之后,V8 就可以依据它们来生成字节码了。AST 之后会被作为输入传到字节码生成器 (BytecodeGenerator),这是 Ignition 解释器中的一部分,用于生成以函数为单位的字节码。[generatedbytecodeforfunction:add(0x079e0824fdc1)]Parametercount3Registercount2Framesize160x79e0824ff7a@0:a7StackCheck0x79e0824ff7b@1:2502Ldara10x79e0824ff7d@3:340300Adda0,[0]0x79e0824ff80@6:26fbStarr00x79e0824ff82@8:0c02LdaSmi[2]0x79e0824ff84@10:26faStarr10x79e0824ff86@12:25fbLdarr00x79e0824ff88@14:abReturnConstantpool(size=0)HandlerTable(size=0)SourcePositionTable(size=0)4、解释执行字节码和 CPU 执行二进制机器代码类似:使用内存中的一块区域来存放字节码;使通用寄存器用来存放一些中间数据;PC 寄存器用来指向下一条要执行的字节码;栈顶寄存器用来指向当前的栈顶的位置。StackCheck 字节码指令就是检查栈是否达到了溢出的上限。Ldar 表示将寄存器中的值加载到累加器中。Add 表示寄存器加载值并将其与累加器中的值相加,然后将结果再次放入累加器。Star 表示 把累加器中的值保存到某个寄存器中。Return 结束当前函数的执行,并将控制权传回给调用方。返回的值是累加器中的值。5、即时编译在解释器 Ignition 执行字节码的过程中,如果发现有热点代码(HotSpot),比如一段代码被重复执行多次,这种就称为热点代码,那么后台的编译器 TurboFan 就会把该段热点的字节码编译为高效的机器码,然后当再次执行这段被优化的代码时,只需要执行编译后的机器码就可以了,这样就大大提升了代码的执行效率。这种字节码配合解释器和编译器的技术被称为即时编译(JIT)。V8 的优化策略下面我们来看一下,V8 为了提升解析和执行 Js 的速度,做了哪些优化。由于篇幅关系,这里只介绍 5 个优化点。1、重新引入字节码早期的 V8 团队认为先生成字节码再执行字节码的方式会降低代码的执行效率,于是直接将 JavaScript 代码编译成机器代码。这样做带来的问题有两点,一是需要较长的编译时间,二是产生的二进制机器码需要占用较大的内存空间。使用字节码的话虽然牺牲了一点执行效率,但是节省了内存空间并且降低了编译时间。此外,字节码也降低了 V8 代码的复杂度,使得 V8 移植到不同的 CPU 架构平台更加容易。这是因为统一将字节码转换为不同平台的二进制代码要比编译器编写不同 CPU 体系的二进制代码更加容易。2、延迟解析通过 V8 的编译流程我们可以看出,V8 执行 JavaScript 代码需要经过编译和执行两个阶段。编译过程:是指 V8 将 JavaScript 代码转换为字节码,或者二进制机器代码的阶段。执行阶段:是指解释器解释执行字节码,或者是 CPU 直接执行二进制机器代码的阶段。V8 并不会一次性将所有的 JavaScript 解析为中间代码,这主要是基于以下两点:如果一次解析和编译所有的 JavaScript 代码,过多的代码会增加编译时间,这会严重影响到首次执行 JavaScript 代码的速度,让用户感觉到卡顿。其次,解析完成的字节码和编译之后的机器代码都会存放在内存中,如果一次性解析和编译所有 JavaScript 代码,那么这些中间代码和机器代码将会一直占用内存。延迟解析是指解析器在解析的过程中,如果遇到函数声明,那么会跳过函数内部的代码,并不会为其生成 AST 和字节码。3、隐藏类我们可以结合一段代码来分析下隐藏类是怎么工作的:letpoint={x:100,y:200}当 V8 执行到这段代码时,会先为 point 对象创建一个隐藏类,在 V8 中,把隐藏类又称为 map,每个对象都有一个 map 属性,其值指向内存中的隐藏类。隐藏类描述了对象的属性布局,它主要包括了属性名称和每个属性所对应的偏移量,比如 point 对象的隐藏类就包括了 x 和 y 属性,x 的偏移量是 4,y 的偏移量是 8。有了隐藏类之后,那么当 V8 访问某个对象中的某个属性时,就会先去隐藏类中查找该属性相对于它的对象的偏移量,有了偏移量和属性类型,V8 就可以直接去内存中取出对应的属性值,而不需要经历一系列的查找过程,那么这就大大提升了 V8 查找对象的效率。4、快属性与慢属性当我们在控制台输入如下代码时:functionFoo(){this[100]='test-100'this[1]='test-1'this["B"]='bar-B'this[50]='test-50'this[9]='test-9'this[8]='test-8'this[3]='test-3'this[5]='test-5'this["A"]='bar-A'this["C"]='bar-C'}varbar=newFoo()for(keyinbar){console.log(`index{key}value{bar[key]}`)}打印出来的结果如下:index:1value:test-1index:3value:test-3index:5value:test-5index:8value:test-8index:9value:test-9index:50value:test-50index:100value:test-100index:Bvalue:bar-Bindex:Avalue:bar-Aindex:Cvalue:bar-C之所以出现这样的结果,是因为在 ECMAScript 规范中定义了数字属性应该按照索引值大小升序排列,字符串属性根据创建时的顺序升序排列。数字属性称为排序属性,在 V8 中被称为 elements。字符串属性就被称为常规属性,在 V8 中被称为 properties。下面我们执行这样一段代码,看一看当对象中的属性数目发生变化时,其在内存中结构是怎样变化的。functionFoo(property_num,element_num){//添加排序属性for(leti=0;i
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 会员注册

本版积分规则

QQ|手机版|心飞设计-版权所有:微度网络信息技术服务中心 ( 鲁ICP备17032091号-12 )|网站地图

GMT+8, 2025-1-16 02:50 , Processed in 0.856082 second(s), 25 queries .

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

快速回复 返回顶部 返回列表