前端 js 之 代码执行的一个过程 02

news2024/12/26 21:40:32

嘿,欢迎你来 !💕

文章目录

  • 前言
  • 一、运行一个 js 文件
  • 二、运行环境
  • 三、js执行代码的过程(普通变量)
  • 四、打印 window
  • 五、js执行代码的过程(函数变量)
  • 六、函数调用函数的过程
  • 注意!!!
  • 最后、作用域与作用域链面试题


前言


在上一篇文章里,我们简单说了浏览器的工作原理和v8引擎的原理



浏览器渲染

  1. 首先先下载 index.html ,遇到css 或者script 文件就下载,
  2. 通过浏览器内核进行解析 index.html
  3. 如果解析过程中遇到 js 代码就立即执行 js 代码,因为js代码属于高级语言,cpu不能直接执行,所以借助js引擎
  4. 对于v8 引擎来说,通过parse模块把我们的代码解析成 AST 树型结构,再通过ignation转化为字节码,再编译成低级语言,cpu就可以执行这些指令了

那今天我们研究的就是 js源代码到AST抽象语法树他进行解析,是如何解析的。


上篇文章整理了很久,多多少少有了点概念,今天这篇文章是基于上篇基础上的加深,希望我表达的足够清晰,如果错处,还望指正,感谢,biubiu ~💕


一、运行一个 js 文件


我们以下的讲解都是基于早期的ECMA的版本规范,也就es5之前的。也发现了所有的变量定义都是用的 var。

在早期的版本规定里是这样定义的:每个执行上下文都会被关联到一个变量环境(VO: variable Object,它是一个对象),源代码中的变量和函数都会 被作为属性 添加到VO中,对于函数来说,参数也会被添加到VO中

在最新的的ECMA 版本规范中,每个执行上下文会关联到一个变量环境中,在执行代码中变量和函数的声明会 作为环境纪录 添加到变量环境中,对于函数来说,参数也会被作为环境记录添加到环境变量中


	  	const name = "haha";
	  	const name2 = "haha2";
	    console.log(name);
	  

假设这是一个 js 文件, 想在浏览器上运行,首先得有一个index.html ,通过script 引入才能运行。js引擎怎么运行的呢?


二、运行环境


我们编写的JS代码都是需要在一个环境中运行的:比如

  1. 浏览器(引擎):例如webkit,gecko,Trident, blink
  2. node*(基于V8渲染js)
  3. webview(v8引擎)
  4. JS之所以能够在浏览器中运行,是因为浏览器给JS提供了执行的环境 => 栈内存

先说几个概念:(不太理解也没关系)

  1. 栈内存 ( ECStack(Execution Context Stack):执行环境栈) :它是代码执行环境;它是浏览器分配的内存空间,用来执行代码 。
  2. 全局对象GO(Globle Object)): 该对象所有作用域都可以访问, 它用来存放当前环境的内置方法,类,变量等,他还有个window属性指向自己;这就是为什么我们可以直接使用Math,String,setTimeout 等方法 ,这里请注意!!对于方法,类这种复杂类型数据存放的是对应的内存地址,堆内存才是真正的存放代码
  3. 堆内存(Heap) :存放东西(属性和方法(例:isNaN:function…))
  4. 任何开辟的内存都有一个16进制的内存地址,方便后期找到这个内存,ox开头
  5. 执行上下文ECS(Execution Context stack) :在编程语言中,代码执行中,为了区分全局变量和函数执行所处的不同的作用域(目的是为了区分每个词法作用域下代码的独立性),会产生执行上下文栈,一个是全局执行上下栈 GEC,一个是函数执行上下文栈FEC,都会在ECS里运行
  6. 全局执行上下栈 GEC: 包括两部分,第一部分,在代码执行前,在paese转化为AST的过程中,会将全局定义的变量,函数等加入到 GO里,但是不会赋值(也称为变量的提升);第二部分,是代码的执行,变量赋值,或者执行函数等等
  7. 函数执行上下文栈FEC:包括三部分,第一部分,在解析函数成为AST树结构时,会创建一个AO,AO中包括形参、arguments、函数定义和指向函数对象、定义的变量;第二部分,作用域,由当前AO对象和父级VO组成,查找是会一层层查找;第三部分,this的绑定值(本篇先不说)

三、js执行代码的过程(普通变量)


  		const name = "haha";
	  	const name2 = "haha2";
	    console.log(name);

	 	 var golbalObject={
	        String:'类',
	        Date:'类',
	        setTimeout:'方法',
	        window:golbalObject,
	        name:undefined,
	        name2:undifined
	      }
  1. 再代码被解析之前,运行之前,v8 引擎内部会帮助我们创建一个对象,全局变量(GlobalObject,又称为GO),
  2. 这个GlobalObject 存放着当前环境的全局对象,该对象所有的作用域都可以访问,
  3. 里面包括 Math,Date,settimeout…等 方法 ,还有一个 window属性 指向自己
  4. 在代码编译的时候,它可以确定我们定义了哪些变量,然后将这些变量添加到全局对象里面
  5. 因为我们还没有真正的执行,所以这些变量都是undefined(变量存放在GO) ,
  6. v8 为了执行代码,v8引擎内部会有一个全局执行上下文栈(全局代码需要被执行的时候才会创建,称为EGC),有且只有一个,EGC 包括两部分: 第一部分,称为VO,在代码执行前,在paese转化为AST的过程中,会将全局定义的变量,函数等加入到 GO里,但是不会赋值(也称为变量的提升),所以这个VO对应的是GO;第二部分,是代码的执行,变量赋值,或者执行函数等等
  7. 比如现在要执行 var name=30 这个代码;它就会去VO里面找,VO对应GO;再把GEC放到调用栈里执行

请添加图片描述


四、打印 window


		const name = "haha";
	  	const name2 = "haha2";
	    console.log(name);

	 	 var golbalObject={
	        String:'类',
	        Date:'类',
	        setTimeout:'方法',
	        window:golbalObject,
	        name:undefined,
	        name2:undifined,
	      }
	     
	    console.log(window)

当打印window时,发现这个对象包括很多东西,很多内置方法,包括name,name2等属性,是因为这个window指向时golbalObject,正如上面所说他一个window属性 指向自己

五、js执行代码的过程(函数变量)


 	 1.   foo();
 	 2.   function foo() {
	 3.      console.log("开心快乐向前冲");
	 4.  }
	  

提前声明 ps:

  1. 我们所有代码在执行前都要加载到内存里,(代码存放在磁盘,再进入内存,转化为机器语言,最后才能被cpu执行)
  2. 在内存里,会划分为栈结构和堆结构,简单类型数据放在栈里,复杂类型数据地址放在栈,值放在堆

运行的不仅仅是普通变量,那对于函数来说,他是一个怎么样的过程呢?

  1. 根据上面所说,在编译阶段,会把变量添加到全局对象里,在未运行之前值为undefined,
  2. 但是foo 在定义之前执行,但是能执行所有的代码,为什么可以打印出来foo呢?
  3. 因为是在编译,所以第一行代码没有执行;第二行发现声明了一个函数,他也会定义 foo在GO里,但是不同的是,因为函数较特殊,他会再开辟一个新的内存空间,用来存放函数, 函数执行上下文栈FEC(先理解这一块,下面会具体说) ,foo这里存放的就是函数对应的内存空间地址
  4. 理解成指针、引用,因为一个地址可以对应多个变量,也是为什么复杂类型数据会互相影响
  5. 以上是编译的过程,当执行到第二行时 foo() ,【请留意,他其实是两部分,foo 和()】,他会再VO (对应的是GO) 里面查找Foo,他发现是个内存地址,就会根据地址找到这个空间;()表示函数的执行,就会把我们 FEC 放入到 调用栈 里面执行。


请添加图片描述



为了不饶晕乎,我们先把以上理解清除,再往下看


	 1.   foo();
	 2.   var name=123
 	 3.   function foo(num) {
 	 4.    	num=1
	 5.     console.log("开心快乐向前冲");
	 6. 	console.log(name)
	 7.  }
  1. 我们把对应的函数空间放到调用栈不是直接运行,而是创建一个函数执行上下文 :包括三部分,第一部分,在解析函数成为AST树结构时,会创建一个AO,AO中包括形参、arguments、函数定义和指向函数对象、定义的变量;第二部分,作用域,由当前AO对象和父级VO组成,查找是会一层层查找;第三部分,this的绑定值(本篇先不说);
  2. 在第六行的时候,我们发现在Ao 里面并没有name这个值,为什么使用了全局的name呢?
  3. 是这样的,当我们查找一个变量时,真实的查找路径是沿着作用域链来查找的
  4. 函数执行上下文不仅仅包括 **VO** ,还有一个 **scope chain** ,叫做作用域链 (看图二)
  5. 一个函数执行上下文的作用域链有两部份组成,当前 AO 的作用域 + 父级作用域
  6. (父级作用域在函数编译时就确定了)
  7. 他的执行顺序是:先在自己的函数执行上下文找,再向他的上级作用域找 所以你理解为什么可以使用name了吗 ?
  8. 一旦函数执行完毕,这个函数执行上下文就会弹出调用栈,就会销毁了,这个 ao 也因为没有指向也会被销毁掉
  9. 当再执行时,是重新创建了一个函数执行上下文,把以上的过程再走一遍



请添加图片描述




六、函数调用函数的过程

我们在上面看了普通变量的执行过程和函数的执行过程,其实发现很简单,调用栈帮我们执行代码,为了区分变量还是函数,我们使用全局执行上下文栈函数执行上下文栈帮助我们执行,编译过程中定义了,真正运行时才是会有赋值等操作。只有函数才有作用域,当我们在一个函数里使用了当前作用域没有定义的变量,而全局变量存在的变量时,也可以使用,是因为沿着作用域链来查找的。在函数定义的时候,他的内存地址里存放两个东西,一个是他的父级作用域和他的执行块

那我们现在看看函数调用函数的过程
其实也很简单,在函数里调用函数时,他发现有函数定义执行时,会再次创建一个函数执行上下文栈,用来存放新的函数,这个函数的VO 同样存储两个东西,一个是作用域(父级作用域和当前作用域)和代码块,如上,所以在函数里调用函数 ,也是按照作用域链查找变量,所以与定义的位置无关

注意!!!


        我们以上的讲解都是基于早期的ECMA的版本规范,也就es5之前的。也发现了所有的变量定义都是用的 var。

在早期的版本规定里是这样定义的: 每个执行上下文都会被关联到一个变量环境(VO: variable Object),再源代码中的变量和函数都会被作为属性添加到VO中,对于函数来说,参数也会被添加到VO中

在最新的ECMA版本规范中,进行了新的改动:每一个执行上下文会关联到一个变量环境中(variableEnvironment),在执行代码中变量和函数的声明会作为环境记录添加到变量环境中,对于函数来说,参数也会被作为环境记录添加到变量环境中

        通过上面的变化,我们可以得知,在最新的ECMA标准中,我们前面的变量对象VO已经有另外一个称呼环境变量,VE



在这里插入图片描述

最后、作用域与作用域链面试题

函数表达式遵循变量提升规则,
今天我的表达还可以吗?🤣

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1109888.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Seata入门系列【10】分布式事务环境下数据库批量插入和批量更新操作

1 前言 批量插入和批量更新是常用的数据库操作,接下来我们分析下在seata 中如何使用。 如果使用循环遍历插入,效率是很慢的,所以一般的ORM框架都是支持批量操作的,接下来以Mybatis 为例,深入了解下如何使用批处理。 …

c语言练习94:分割链表

分割链表 给你一个链表的头节点 head 和一个特定值 x ,请你对链表进行分隔,使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。 你不需要 保留 每个分区中各节点的初始相对位置。 示例 1: 输入:head [1,4,3,2,5,2], x…

python小游戏:打砖块完整代码

代码如下 import pygame import sys# 初始化Pygame pygame.init()# 游戏窗口尺寸 WIDTH, HEIGHT 800, 600# 颜色定义 WHITE (255, 255, 255) BLUE (0, 0, 255) GREEN (0, 255, 0)# 创建游戏窗口 win pygame.display.set_mode((WIDTH, HEIGHT)) pygame.display.set_captio…

2023年高校大数据实验室建设及大数据实训平台整体解决方案

大数据实验室作为支撑高校人培方案实施的核心设施,大数据实训实验室的建设一定要与学科建设、人才培养充分融合,是一个包含物理空间硬件资源软件资源课程内容的系统化工程。高校大数据实验室建设,除了考虑物理空间与硬件资源外,重…

实验室超声波清洗机的作用

实验室超声波清洗机的作用是什么?顾名思义,其主要作用是清洗。超声波清洗机是实验室中必不可少的清洗装置,利用超声波在液体中的空化效应,产生空化气泡,由于正负压的作用下,空化气泡会在短时间内生成并爆破…

移动协同办公系统,让工作更轻松,生活更美好!

在今天这个信息化、数字化的时代,人们对于工作效率的需求越来越高。随着移动互联网的普及,越来越多的人开始将工作与生活的界限模糊化。在这个背景下,移动协同办公系统应运而生,它打破了传统办公模式的束缚,让工作更轻…

通讯协议学习之路:QSPI协议理论

通讯协议之路主要分为两部分,第一部分从理论上面讲解各类协议的通讯原理以及通讯格式,第二部分从具体运用上讲解各类通讯协议的具体应用方法。 后续文章会同时发表在个人博客(jason1016.club)、CSDN;视频会发布在bilibili(UID:399951374) 一、…

【动态规划】392. 判断子序列、115. 不同的子序列

提示:努力生活,开心、快乐的一天 文章目录 392. 判断子序列💡解题思路🤔遇到的问题💻代码实现🎯题目总结 115. 不同的子序列💡解题思路🤔遇到的问题💻代码实现&#x1f3…

进程地址空间初识

进程地址空间: 空间布局图: 下面来看这样一段代码: 对应执行的结果如下: 可以看到一个很奇怪的现象: g_val的值不同,却用的是同一块地址 这是为什么? 说明对应的地址一定不是物理地址&#…

EF执行迁移时提示provider: SSL Provider, error: 0 - 证书链是由不受信任的颁发机构颁发的

ef在执行时提示provider: SSL Provider, error: 0 - 证书链是由不受信任的颁发机构颁发的。 只需要在数据库链接字符串后增加EncryptTrue;TrustServerCertificateTrue;即可 再次执行

c++生成随机数(猜数字游戏)

如果之前学过python的小伙伴看到随机数这个知识点肯定会觉得很简单,但是c的随机数就需要注意一些小细节了~ 先来给大家展示一下python的随机数: import random #导入随机数库 arandom.randint(100,200)#随机100-200之间的数字 以下是c的&a…

【黑夜送书第一期】好书来袭,AI时代程序员/项目经理开发之道送3本~

🏆作者简介,黑夜开发者,CSDN领军人物,全栈领域优质创作者✌,CSDN博客专家,阿里云社区专家博主,2023年6月CSDN上海赛道top4。 🏆数年电商行业从业经验,历任核心研发工程师…

Elasticsearch:生成式人工智能带来的社会转变

作者:JEFF VESTAL 了解 Elastic 如何走在大型语言模型革命的最前沿 – 通过提供实时信息并将 LLM 集成到数据分析的搜索、可观察性和安全系统中,帮助用户将 LLM 提升到新的高度。 iPhone 社会转变:新时代的黎明 曾几何时,不久前…

毫米波雷达2-雷达的工作模式

文章目录 flash mode: 用于烧写functional mode: Power off the board and remove the jumper from only header SOP2 (this puts the board back in functional mode) flash mode: 用于烧写 functional mode: Power off the board and remove the jumper from only header SOP…

配电房环境智能监控系统:守护电力设施,保障安全运行

随着电力系统的日益复杂化,配电房的环境监控显得尤为重要。传统的监控方式已经无法满足现代配电房的需求,因此,配电房环境智能监控系统应运而生。 一、系统概述 力安科技电易云配电房环境智能监控系统是一款集成了环境参数监测、设备运行…

当AI遇上3D建模:一场创意与技术的完美碰撞!

在这个充满无限可能的时代,科技的发展日新月异,让我们见证了一场创意与技术的完美碰撞——AI与3D建模的结合。这种结合为我们开辟了一个全新的领域,让我们在虚拟世界中自由挥洒创意,为现实世界创造更多的可能性。 3D建模AI生成是…

Windows下定时下载Linux服务器的数据库备份文件(pscp+bat脚本+定时任务)

下载传输软件pscp Download PuTTY: latest release (0.79) 创建bat执行脚本 echo 删除旧的备份文件 del D:\db_bk\*.dbecho 下载新的备份文件 D:\Programs\pscp -P 22 -pw youPassword youName192.168.1.1:/home/backup/test.db D:\db_bk\ 设置定时任务 1.使用任务计划程…

2023年【天津市安全员C证】报名考试及天津市安全员C证操作证考试

题库来源:安全生产模拟考试一点通公众号小程序 天津市安全员C证报名考试是安全生产模拟考试一点通生成的,天津市安全员C证证模拟考试题库是根据天津市安全员C证最新版教材汇编出天津市安全员C证仿真模拟考试。2023年【天津市安全员C证】报名考试及天津市…

代码与细节(一)

在用到 Java17的新特性 Unmodifiable Lists 时不知道你是否和我有同样的惊讶 为什么弄了这么多重载方法? 先说结论:为了性能。 其实一细想,都能想明白:varargs(可变参数) 的背后是数组的内存分配和初始化,相比正常的…

36张图详解网络基础知识

下午好,我的网工朋友。 在网工这行,只要是面试,一般都难逃网络协议相关的问题吧。 不管是OSI还是TCP/IP,这都是非常重要、基础的知识,很多知识点都是以它们为基础去串联的。 作为网络世界的底层技术,掌握…