一、概念:事件循环

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. 集训队8月2日(BFS)

    看书情况:109~124页 刷题数:6 今天把上两次比赛的该补的题都补了,补题有博客,还写了两道书上例题的博客. 书上例题 BFS思维https://www.cnblogs.com/246247839 ...

  2. Git Git管理码云项目

    Git  一.下载安装 1. 要使用git 先安转git 请到官网下载最新git  https://git-scm.com/downloads 2. 一路默认安装,安装完成右键查看下是否有Git. 二 ...

  3. php开发面试题---jquery和vue对比(整理)

    php开发面试题---jquery和vue对比(整理) 一.总结 一句话总结: jquery的本质是更方便的选取和操作DOM对象,vue的本质是数据和页面分离 反思的回顾非常有用,因为决定了我的方向和 ...

  4. What size do you use for varchar(MAX) in your parameter declaration?

    What size do you use for varchar(MAX) in your parameter declaration? In this case you use -1. See al ...

  5. 如何用python读写CSV 格式文件

    工作中经常会碰到读写CSV文件的情况.记录下,方便自己以后查询并与大家一起分享: 写CSV文件方法一: import csv          #导入CSV with open("D:\eg ...

  6. 77、tensorflow手写识别基础版本

    ''' Created on 2017年4月20日 @author: weizhen ''' #手写识别 from tensorflow.examples.tutorials.mnist import ...

  7. Django框架(十四)—— Django分页组件

    目录 Django分页组件 一.分页器 二.分页器的使用 三.案例 1.模板层 2.视图层 Django分页组件 一.分页器 数据量大的话,可以分页获取,查看 例如:图书管理中,如果有成千上万本书,要 ...

  8. Spring JAR下载地址

    包含3.2版本及以上 http://repo.spring.io/libs-release-local/org/springframework/spring/ 包含从2.0开始的所有版本 http:/ ...

  9. 转 Nginx Access Log日志统计分析常用命令

    Nginx Access Log日志统计分析常用命令Nginx Access Log日志统计分析常用命令IP相关统计 统计IP访问量 awk '{print $1}' access.log | sor ...

  10. libvirt虚拟机管理常用指令

    创建虚拟机 virt-install virt-install --connect qemu:///system -n $NAME -r $MEM -f $DISK -s $DISK_SIZE --v ...