一、概念:事件循环

JavaScript是单线程的

1、整片 script 整体代码(第一个宏任务)放到执行栈中,执行之后,会触发很多方法

这些方法只能一个个的顺序执行,不能并发

2、这些要执行的方法会放到一个称之为事件队列的地方

3、事件队列又分为宏任务队列和微任务队列,所以要执行的方法会被分类到各自类型的队列列表

4、微任务队列的任务是最优先的,只有微任务队列的任务执行完成后,才会去宏任务队列取队头的第一个任务放到执行栈执行

5、之后可能又产生宏任务和微任务,继续步骤4,微任务优先完成,不断循环下去直到所有任务队列的任务清空

这是整体的一个流程

二、分类

下面分类介绍下:

术语:

1)、执行栈:用来执行当前代码

例子:

各个同步代码

a()

b()

c()

2)、宏任务队列:存储宏任务任务

例子:

script 整体代码

setInterval

setTimeout

I/O

UI 交互

setImmediate

3)、微任务队列:存储微任务任务

例子:

Promise.then

process.nextTick

4)、异步请求未完成挂起队列:存储异步耗时任务

例如:未完成的 ajax 请求被挂起

三、流程:

浏览器解析脚本---

解析到的同步代码按照顺序推入执行栈中---

1、顺序执行执行栈中的所有同步代码---

2、执行完执行栈中的所有同步代码,产生 宏任务队列 微任务队列 异步请求未完成挂起队列---

3、此时执行栈为空

4、查看微任务队列是否有任务?

5、有微任务则执行!当次微任务队列全部执行完,微任务队列为空,才去查看宏任务队列

6、没有微任务则直接查看宏任务队列是否有任务,有,取出宏任务队列最前面的任务推入执行栈---

注:在执行微任务宏任务的过程中,异步请求未完成挂起队列中的任务可能已经完成,完成后会推入宏任务队列

7、顺序执行执行栈中的所有同步代码---如此循环

四、练习例子

例子一

// 请写出输出内容 --- 网上搜出来的一个例子
async function async1() {
console.log('async1 start') // 2
await async2()
console.log('async1 end') // 6
}
async function async2() {
console.log('async2') // 3
} console.log('script start') // 1 setTimeout(function() {
console.log('setTimeout') // 8
}, 0) async1() new Promise(function(resolve) {
console.log('promise1') // 4
resolve()
}).then(function() {
console.log('promise2') // 7
})
console.log('script end') // 5 // 输出结果:
// script start
// async1 start
// async2
// promise1
// script end
// async1 end
// promise2
// setTimeout 解析:
第一轮:解析整个代码片段,相当于第一个宏任务在执行栈中执行
此时状态:
执行栈:整片代码段
宏任务队列:null
微任务队列:null 执行过程:
1、遇到 console.log('script start') 直接执行
2、遇到 setTimeout(function() {
console.log('setTimeout')
}, 0)
获得一个宏任务
`function() {
console.log('setTimeout')
}`
3、遇到 async1(),执行后
解析到 console.log('async1 start') 直接执行
然后解析到 await async2()
执行 async2() 后,遇到 console.log('async2') 直接执行
上面执行完成后,await 之后的动作作为一个微任务 `console.log('async1 end')` 被收集
4、遇到 new Promise
直接执行 console.log('promise1'),然后获取一个微任务 `console.log('promise2')`
5、最后直接执行 console.log('script end') 执行后状态:
执行栈:null
宏任务队列:
`function() {
console.log('setTimeout')
}`
微任务队列
`console.log('async1 end')`
`console.log('promise2')` 第二轮:查看执行微任务队列,然后是宏任务队列
此时状态(即上一轮执行后的状态):
执行栈:null
宏任务队列:
`function() {
console.log('setTimeout')
}`
微任务队列
`console.log('async1 end')`
`console.log('promise2')` 执行过程:
1、先看微任务队列是否有任务,先清空微任务队列
将 console.log('async1 end') 拿到执行栈执行
执行完成后 将 console.log('promise2') 拿到执行栈执行
执行完成,执行栈此时为空
2、此时微任务队列都执行完成了,执行栈也为空,从宏任务队列队头取第一个任务
将 function() {
console.log('setTimeout')
} 拿到执行栈执行
执行完成,所有任务都清空,程序结束

### 例子二
```javascript
setTimeout(() =>{
console.log(1)
setTimeout(() =>console.log(2))
setTimeout(() =>console.log(3))
Promise.resolve().then(() => console.log('a'))
.then(() => console.log('b'))
.then(() => console.log('c'))
.then(() => console.log('d'))
})
// 输出结果:1 a b c d 2 3

解析:

1、执行整片代码,产生一个宏任务,没有微任务

宏任务:

() =>{ console.log(1) setTimeout(() =>console.log(2)) setTimeout(() =>console.log(3)) Promise.resolve().then(() => console.log('a')) .then(() => console.log('b')) .then(() => console.log('c')) .then(() => console.log('d')) .then(() => console.log('e')) }

2、将宏任务拿到执行栈执行,陆续执行里面的代码

产生两个宏任务和一个微任务

宏任务:

() =>console.log(2)

() =>console.log(3)

微任务:

() => console.log('a')

3、将一个微任务拿到执行栈执行

此时又产生一个微任务

宏任务:

() =>console.log(2)

() =>console.log(3)

微任务:

() => console.log('b')

4、将一个微任务拿到执行栈执行

此时又产生一个微任务

宏任务:

() =>console.log(2)

() =>console.log(3)

微任务:

() => console.log('c')

5、将一个微任务拿到执行栈执行

此时又产生一个微任务

宏任务:

() =>console.log(2)

() =>console.log(3)

微任务:

() => console.log('d')

6、将一个微任务拿到执行栈执行

微任务执行完成,取宏任务队列队头第一个任务

7、执行 () =>console.log(2) 完后,没有产生微任务,取第二个任务

8、执行 () =>console.log(3) 完后,没有任务,程序结束

聊一聊JavaScript中的事件循环的更多相关文章

  1. 深入理解javascript中的事件循环event-loop

    前面的话 本文将详细介绍javascript中的事件循环event-loop 线程 javascript是单线程的语言,也就是说,同一个时间只能做一件事.而这个单线程的特性,与它的用途有关,作为浏览器 ...

  2. JavaScript中的事件循环机制跟函数柯里化

    一.事件循环机制的理解 test();//按秒输出5个5 function test() { for (var i = 0; i < 5; i++) { setTimeout(() => ...

  3. JavaScript中的事件循环

    JavaScript是单线程单并发语言 单线程:主程序只有一个线程,即同一时间片段内其只能执行单个任务. 引发的问题: 单线程,意味着任务都需要排队,前一个任务结束,才会执行后一个任务.若前一个任务耗 ...

  4. javascript 中的事件机制

    1.javascript中的事件. 事件流 javascript中的事件是以一种流的形式存在的. 一个事件会也有多个元素同时响应. 有时候这不是我们想要的效果, 我们只是需要某个特定的元素相应我们的绑 ...

  5. JavaScript中的事件对象

    JavaScript中的事件对象 JavaScript中的事件对象是非常重要的,恐怕是我们在项目中使用的最多的了.在触发DOM上的某个事件时,会产生一个事件对象event,这个对象中包含这所有与事件有 ...

  6. JavaScript 进阶教程一 JavaScript 中的事件流 - 事件冒泡和事件捕获

    先看下面的示例代码: <html xmlns="http://www.w3.org/1999/xhtml"> <head> <title>Jav ...

  7. 了解javascript中的事件(二)

    本文目录如下: 零.寒暄 一.事件的分类 二.事件代理 2.1 问题引出 2.2 什么是事件代理 2.3 完整示例 二.事件代理 三.事件代理思想的用处 四.总结 零.寒暄 这篇博客本该出现在两个月以 ...

  8. javascript中onclick事件能调用多个方法吗

    Q: javascript中onclick事件能调用多个方法吗? A: 可以的,方法如下onclick="aa();bb();cc();"每个方法用“;”分号隔开就行了

  9. js实例分析JavaScript中的事件委托和事件绑定

    我们在学习JavaScript中,难免都会去网上查一些资料.也许偶尔就会遇到“事件委托”(也有的称我“事件代理”,这里不评论谁是谁非.以下全部称为“事件委托”),尤其是在查JavaScript的事件处 ...

随机推荐

  1. django 304

    pycharm现象: [18/Jul/2017 23:27:26] "GET /static/201408210521231921042235.png HTTP/1.1" 304 ...

  2. Spark核心原理初探

    一.运行架构概览 Spark架构是主从模型,分为两层,一层管理集群资源,另一层管理具体的作业,两层是解耦的.第一层可以使用yarn等实现. Master是管理者进程,Worker是被管理者进程,每个W ...

  3. Mysql 列变行其中一种做法。

    需求是: 上班打卡记录   和  下班打卡记录  是分别是两条数据,现在是要合并为一条数据,并且封装成一个实体. 有可能是 只有上班记录,,或者是只有下班的记录.如何关联全查,一边为null或者另一边 ...

  4. MariaDB 管理

    在尝试运行MariaDB之前,首先确定其当前状态,运行或关闭. 有三个选项用于启动和停止MariaDB - 运行mysqld(MariaDB脚本). 运行mysqld_safe启动脚本. 运行mysq ...

  5. 为什么不能在shell脚本中执行source /etc/profile或者source ~/.bashrc问题?

    执行脚本时,其中的命令是在一个子shell中执行的.子shell继承了父shell的环境变量,但无法修改他们,或者说所做的修改仅对子shell有效.

  6. ROM、PROM、EPROM、EEPROM、Flash ROM分别指什么?

    ROM指的是“只读存储器”,即Read-Only Memory.这是一种线路最简单半导体电路,通过掩模工艺, 一次性制 造,其中的代码与数据将永久保存(除非坏掉),不能进行修改.这玩意一般在大批量生产 ...

  7. 前端工程师技能图谱skill-map

    # 前端工程师技能图谱 ## 浏览器 - IE6/7/8/9/10/11 (Trident) - Firefox (Gecko) - Chrome/Chromium (Blink) - Safari ...

  8. oracle使用execute immediate方式完成函数动态传入表名并操作 返回新的主键id值

    CREATE OR REPLACE FUNCTION SEQ1 (v_bname in VARCHAR2) return NUMBER is v_bcount NUMBER; BEGIN execut ...

  9. 2019杭电多校第六场hdu6638 Snowy Smile(线段树+枚举)

    Snowy Smile 题目传送门 解题思路 先把y离散化,然后把点按照x的大小进行排序,我们枚举每一种x作为上边界,然后再枚举其对应的每一种下边界.按照这种顺序插入点,这是一个压维的操作,即在线段树 ...

  10. python2和python3中int整型数据的不同之处

    python2中的除法,结果为整型数字(int型),例如 10/5=2,10/3=3,小数位向下取整 python3中的除法,结果为浮点型数字(float型)结果小数位最多保留16位小数