JS EventLoop 动图学习 (一)

JavaScript是单线程的语言

事件循环Event Loop,这是目前浏览器和NodeJS处理JavaScript代码的一种机制,而这种机制存在的背后,就有因为JavaScript是一门单线程的语言。

单线程和多线程最简单的区别就是:单线程同一个时间只能做一件事情,而多线程同一个时间能做多件事情。

JavaScript之所谓设计为单线程语言,主要是因为它作为浏览器脚本语言,主要的用途就是与用户互动,操作Dom节点。

而在这个情景设定下,假设JavaScript同时有两个进程,一个是操作A节点,一个是删除A节点,这时候浏览器就不知道要以哪个线程为准了。

因此为了避免这类型的问题,JavaScript从一开始就属于单线程语言。

调用栈 Call Stack

JavaScript运行的时候,主线程会形成一个栈,这个栈主要是解释器用来最终函数执行流的一种机制。通常这个栈被称为调用栈Call Stack,或者执行栈(Execution Context Stack)。

调用栈,顾名思义是具有LIFO(后进先出,Last in First Out)的结构。调用栈内存放的是代码执行期间的所有执行上下文。

  • 每调用一个函数,解释器就会把该函数的执行上下文添加到调用栈并开始执行;
  • 正在调用栈中执行的函数,如果还调用了其他函数,那么新函数也会被添加到调用栈,并立即执行;
  • 当前函数执行完毕后,解释器会将其执行上下文清除调用栈,继续执行剩余执行上下文中的剩余代码;
  • 但分配的调用栈空间被占满,会引发”堆栈溢出“的报错。

现在用个小案例来演示一下调用栈。

function a() {
    console.log('a');
}

function b() {
    console.log('b');
}

function c() {
    console.log('c');
    a();
    b();
}

c();

/**
* 输出结果:c a b
*/

执行这段代码的时候,首先调用的是函数c()。因此function c(){}的执行上下文就会被放入调用栈中。

在这里插入图片描述
然后开始执行函数c,执行的第一个语句是console.log('c')

因此解释器也会将其放入调用栈中。

在这里插入图片描述
console.log('c')方法执行完后,控制台打印了'c',调用栈就会将其移除。

在这里插入图片描述
接着就是执行a()函数。

解释器就将function a() {}的执行上下文放入调用栈中。

在这里插入图片描述
紧接着就执行a()中的语句——console.log('a')

在这里插入图片描述
当函数a执行结束后,调用栈就将执行上下文移除。

然后接着执行c()函数剩下的语句,也就是执行b()函数,因此它的执行上下文就加入调用栈中。

在这里插入图片描述
紧接着就执行b()中的语句——console.log('b')

在这里插入图片描述

b()执行完后,调用栈就将其移出。

这时c()也执行结束了,调用栈也将其移出栈。

在这里插入图片描述
这时候,我们这段语句就执行结束了。

任务队列

上面的案例简单的介绍了关于JavaScript单线程的执行方式。

但这其中会存在一些问题,就是如果当一个语句也需要执行很长时间的话,比如请求数据、定时器、读取文件等等,后面的语句就得一直等着前面的语句执行结束后才会开始执行。

显而易见,这是不可取的。

同步任务和异步任务

因此,JavaScript将所有执行任务分为了同步任务和异步任务。

其实我们每个任务都是在做两件事情,就是发起调用得到结果

而同步任务和异步任务最主要的差别就是,同步任务发起调用后,很快就可以得到结果,而异步任务是无法立即得到结果,比如请求接口,每个接口都会有一定的响应时间,根据网速、服务器等等因素决定,再比如定时器,它需要固定时间后才会返回结果。

因此,对于同步任务和异步任务的执行机制也不同。

同步任务的执行,其实就是跟前面那个案例一样,按照代码顺序和调用顺序,支持进入调用栈中并执行,执行结束后就移除调用栈。

而异步任务的执行,首先它依旧会进入调用栈中,然后发起调用,然后解释器会将其响应回调任务放入一个任务队列,紧接着调用栈会将这个任务移除。当主线程清空后,即所有同步任务结束后,解释器会读取任务队列,并依次将已完成的异步任务加入调用栈中并执行。

这里有个重点,就是异步任务不是直接进入任务队列的。

这里举一个简单的例子。

console.log(1);

fetch('https://jsonplaceholder.typicode.com/todos/1')
    .then(response => response.json())
    .then(json => console.log(json))

console.log(2);

但执行到console.log(2)之前,其实fetch()已经被调用且发起请求了,但是还未响应数据。而响应数据和处理数据的函数then()此时已经在任务队列中,等候console.log(2)执行结束后,所以同步任务清空后,再进入调用栈执行响应动作。

在这里插入图片描述

宏任务和微任务

前面聊到同步任务和异步任务的时候,提及到了任务队列

在任务队列中,其实还分为宏任务队列(Task Queue)微任务队列(Microtask Queue),对应的里面存放的就是宏任务微任务

首先,宏任务和微任务都是异步任务

而宏任务和微任务的区别,就是它们执行的顺序,这也是为什么要区分宏任务和微任务。

在同步任务中,任务的执行都是按照代码顺序执行的,而异步任务的执行也是需要按顺序的,队列的属性就是先进先出(FIFO,First in First Out),因此异步任务会按照进入队列的顺序依次执行。

但在一些场景下,如果只按照进入队列的顺序依次执行的话,也会出问题。比如队列先进入一个一小时的定时器,接着再进入一个请求接口函数,而如果根据进入队列的顺序执行的话,请求接口函数可能需要一个小时后才会响应数据。

因此浏览器就会将异步任务分为宏任务和微任务,然后按照事件循环的机制去执行,因此不同的任务会有不同的执行优先级,具体会在事件循环讲到。

任务入队

这里还有一个知识点,就是关于任务入队。

任务进入任务队列,其实会利用到浏览器的其他线程。虽然说JavaScript是单线程语言,但是浏览器不是单线程的。而不同的线程就会对不同的事件进行处理,当对应事件可以执行的时候,对应线程就会将其放入任务队列。

  • js引擎线程:用于解释执行js代码、用户输入、网络请求等;
  • GUI渲染线程:绘制用户界面,与JS主线程互斥(因为js可以操作DOM,进而会影响到GUI的渲染结果);
  • http异步网络请求线程:处理用户的get、post等请求,等返回结果后将回调函数推入到任务队列;
  • 定时触发器线程setIntervalsetTimeout等待时间结束后,会把执行函数推入任务队列中;
  • 浏览器事件处理线程:将clickmouse等UI交互事件发生后,将要执行的回调函数放入到事件队列中。

这个其实就可以解释了下列代码为什么后面的定时器会比前面的定时器先执行。因为后者的定时器会先被推进宏任务队列,而前者会之后到点了再被推入宏任务队列。

setTimeout(() => {
   console.log('a');
}, 10000);

setTimeout(() => {
   console.log('b');
}, 100);
宏任务
名称浏览器Node
整体代码(script)
UI交互事件
I/O
setTimeout
setInterval
setImmediate
requestAnimationFrame
微任务
名称浏览器Node
process.nextTick
MutationObserver
Promise.then catch finally

以上就是异步任务宏任务微任务的分类;

事件循环 Event Loop

其实宏任务队列和微任务队列的执行,就是事件循环的一部分了,所以放在这里一起说。

事件循环的具体流程如下:

  • 1、从宏任务队列中,按照入队顺序,找到第一个执行的宏任务,放入调用栈,开始执行;
  • 2、执行完该宏任务下所有同步任务后,即调用栈清空后,该宏任务被推出宏任务队列,然后微任务队列开始按照入队顺序,依次执行其中的微任务,直至微任务队列清空为止
  • 3、当微任务队列清空后,一个事件循环结束;
  • 4、接着从宏任务队列中,找到下一个执行的宏任务,开始第二个事件循环,直至宏任务队列清空为止。

这里有几个重点:

  • 当我们第一次执行的时候,解释器会将整体代码script放入宏任务队列中,因此事件循环是从第一个宏任务开始的;
  • 如果在执行微任务的过程中,产生新的微任务添加到微任务队列中,也需要一起清空;微任务队列没清空之前,是不会执行下一个宏任务的。

接下来,通过一个常见的面试题例子来模拟一下事件循环。

console.log("a");

setTimeout(function () {
    console.log("b");
}, 0);

new Promise((resolve) => {
    console.log("c");
    resolve();
})
    .then(function () {
        console.log("d");
    })
    .then(function () {
        console.log("e");
    });

console.log("f");

/**
* 输出结果:a c f d e b
*/

首先,当代码执行的时候,整体代码script被推入宏任务队列中,并开始执行该宏任务。

在这里插入图片描述
按照代码顺序,首先执行console.log("a")

该函数上下文被推入调用栈,执行完后,即移除调用栈。

在这里插入图片描述
接下来执行setTimeout(),该函数上下文也进入调用栈中。

在这里插入图片描述
因为setTimeout是一个宏任务,因此将其callback函数推入宏任务队列中,然后该函数就被移除调用栈,继续往下执行。

在这里插入图片描述
紧接着是Promise语句,先将其放入调用栈,然后接着往下执行。

在这里插入图片描述

执行console.log("c")resolve(),这里就不多说了。

在这里插入图片描述
接着来到new Promise().then()方法,这是一个微任务,因此将其推入微任务队列中。

在这里插入图片描述

这时new Promise语句已经执行结束了,就被移除调用栈。

接着做执行console.log('f')

在这里插入图片描述

这时候,script宏任务已经执行结束了,因此被推出宏任务队列。

紧接着开始清空微任务队列了。首先执行的是Promise then,因此它被推入调用栈中。

在这里插入图片描述

然后开始执行其中的console.log("d")

执行结束后,检测到后面还有一个then()函数,因此将其推入微任务队列中。

此时第一个then()函数已经执行结束了,就会移除调用栈和微任务队列。

在这里插入图片描述
此时微任务队列还没被清空,因此继续执行下一个微任务。

执行过程跟前面差不多,就不多说了。

在这里插入图片描述
此时微任务队列已经清空了,第一个事件循环已经结束了。

接下来执行下一个宏任务,即setTimeout callback

在这里插入图片描述
执行结束后,它也被移除宏任务队列和调用栈。

这时候微任务队列里面没有任务,因此第二个事件循环也结束了。

宏任务也被清空了,因此这段代码已经执行结束了。

在这里插入图片描述

今天先到这里。接下来有空再合并第二期

2022.6.3日:打个广告,苏州华为终端BG面向社会招聘人才,Java /C C++ / Python / Javascript 。有兴趣来苏州的同学们 可以加我V 15850277051 ,走内推流程,有问必答!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值