iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > JavaScript >实例详解JS中的事件循环机制
  • 126
分享到

实例详解JS中的事件循环机制

2024-04-02 19:04:59 126人浏览 泡泡鱼
摘要

目录一、前言二、宏、微任务三、Tick 执行顺序四、案例详解1.掺杂setTimeout2.掺杂微任务,此处主要是Promise.then3.掺杂async/await一、前言 之前

一、前言

之前我们把React相关钩子函数大致介绍了一遍,这一系列完结之后我莫名感到空虚,不知道接下来应该更新有关哪方面的文章。最近想了想,打算先回归一遍js基础,把一些比较重要的基础知识点回顾一下,然后继续撸框架(可能是源码、也可能补全下全家桶)。不积跬步无以至千里,万丈高楼咱们先从JS的事件循环机制开始吧,废话不多说,开搞开搞!

在JS中,我们所有的任务可以分为同步任务和异步任务。那么什么是同步任务?什么又是异步任务呢?

同步任务:是在主线程执行栈上排队执行的任务,只有前一个任务执行完毕,才能执行后一个任务;比如:console.log、赋值语句等。

异步任务:不进入主线程,是进入任务队列的任务,只有等主线程任务执行完毕,"任务队列"开始通知主线程,请求执行任务,该任务才会进入主线程执行。比如:ajax网络请求,setTimeout 定时函数等都属于异步任务,异步任务会通过任务队列的机制(先进先出的机制)来进行协调。

我们执行一段代码时,在我们主线程的执行栈执行过程中,如果遇到同步任务会立即执行,如果遇到异步任务会暂时挂起,将此异步任务推入任务队列中(队列的执行机制遵循先进先出)。当主线程执行栈里的同步任务执行完毕后,js执行引擎会去任务队列中读取挂起的异步任务并将其推入到执行栈中执行。这个不断重复的过程(执行栈执行--->判断同异步--->同步执行/异步挂起推入事件对列--->栈空后取事件队列里任务并推入执行栈执行--->继续判断同异步--->.......)就是本文所要介绍的事件循环。

二、宏、微任务

我们每进行一次事件循环的操作被称之为tick,在介绍一次 tick 的执行步骤之前,我们需要补充两个概念:宏任务、微任务。

宏任务和微任务严格来说是es6之后才有的概念(原因在于ES6提出了Promise这个概念);在Es6之后我们把JS的任务更细分成了宏任务和微任务。

其中,宏任务主要包括:script(整体代码)、setTimeout、setInterval、I/O、UI交互事件、postMessage、requestAnimationFrame(帧动画)、MessageChannel、setImmediate(node.js环境);

微任务主要包括:Promise.then、MutaionObserver、process.nextTick(node.js环境);

好了,了解了宏微任务的概念之后我们就来掰扯掰扯每次tick的执行顺序吧。首先看下图:

三、Tick 执行顺序

1、首先执行一个宏任务(栈中没有就从事件队列中获取);

2、执行过程中如果遇到微任务,就将它添加到微任务的任务队列中、如果有宏任务的话推到相应的事件队列中去;

3、宏任务执行完毕后,立即执行当前微任务队列中的所有微任务(依次执行);

4、当前宏任务执行完毕,开始进行渲染;5、开始下一个宏任务(从事件队列中获取)开启下一次的tick;

需要注意的是:宏任务执行过程中如果宏任务中又添加了一个新的宏任务到任务队列中。 这个新的宏任务会等到下一次事件循环再执行;而微任务则不同,微任务执行过程中如果又添加了新的微任务,则新的微任务也会在本次微任务执行过程中被执行,直到微任务队列为空。每次宏任务执行完在开启下一次宏任务时会把微任务队列中所有的微任务执行完毕!

四、案例详解

概念性的东西说完了,下面就来找些demo练练手吧!

1.掺杂setTimeout

console.log('开始');

setTimeout(()=>{
    console.log('同级的定时器');
      setTimeout(() => {
          console.log('内层的定时器');
      }, 0);
},0)

console.log('结束');

输出结果为

开始 -> 结束 -> 同级的定时器 ->内层的定时器

解释上述代码:

  • 整体代码作为一个宏任务进入主线程执行栈中;
  • 遇到console.log('开始'),控制台输出 开始;
  • 遇到有一个宏任务setTimeout,JS引擎将之挂起,并推入任务队列;
  • 遇到console.log('结束'),控制台输出 结束;本次宏任务执行完毕,发现本次并无微任务,GUI进行render渲染完毕开启下一次宏任务执行,本次tick结束。
  • JS引擎从任务队列拿出第一个setTimeout宏任务,将至推入主线程执行栈, 开始进行第二个宏任务;
  • 执行setTimeout回调,遇到 console.log('同级的定时器'),控制台输出 同级的定时器;
  • 遇到第二个setTimeout ,这是个本次宏任务产生的新的宏任务,将此宏任务挂起,并推入任务队列;
  • 同样此时发现没有微任务,则GUI接管开始进行渲染,渲染完毕又开启下一次宏任务,tick结束;
  • JS引擎又从任务队列拿出第二个setTimeout宏任务,将之推入主线程执行栈, 开始进行第三个宏任务;
  • 执行第二个setTimeout回调,遇到 console.log('内层的定时器'),控制台输出 内层的定时器;
  • 本次宏任务执行完毕发现没有微任务,结束。

2.掺杂微任务,此处主要是Promise.then

console.log('script start');

setTimeout(function() {
  new Promise(resolve=>{
        console.log('000');
      resolve()
    }).then(res=>{
        console.log('这是微任务');
    })
  console.log('timeout1');
}, 10);

new Promise(resolve => {
    console.log('promise1');       
    resolve();
    setTimeout(() => console.log('timeout2'), 10);
}).then(function() {
    console.log('then1')
})

console.log('script end');

输出结果为:

script start -> promise1 -> script end -> then1 -> 000 -> timeout1 -> 这是微任务 -> timeout2

解释上述代码:

  • 整体script作为一个宏任务进入主线程执行栈;
  • 遇到 console.log('script start') 输出 script start ;
  • 遇到setTimeout作为新的一个宏任务连同其回调内容一同推入任务队列 ;
  • 遇到和script start 同级的new Promise 进行执行,此处需注意:Promise内容是同步任务,它的.then才是微任务会被推入微任务队列。所有此处JS引擎的处理逻辑是:遇到 console.log('promise1') 输出 promise1 ,遇到resolve() 会将 Promise的.then函数推入微任务队列(注意,我们常说微任务时宏任务的小尾巴,指的是本次宏任务产生的微任务都会在本次宏任务执行完之后进行执行清空。);遇到resolve下面的setTimeout这是个新的宏任务,会被挂起并推入任务队列。
  • 继续顺序执行,执行到 console.log('script end') ,输出script end;此时第一个宏任务执行完毕,JS引擎开始清理小尾巴(执行并清空微任务队列)。
  • 此时由本次执行宏任务的过程中产生了 .then(function() { console.log('then1')}) 这个微任务,JS引擎会将此任务内的回调推入执行栈进行执行,输出 then1;
  • 微任务队列为空,开启下一个宏任务,第一轮tick结束;
  • JS引擎从任务队列中拿script start下面那个setTimeout宏任务将回调推入主线程执行栈中进行执行;
  • 遇到了Promise,执行其内容:遇到 console.log('000') 输出 000;
  • 执行 resolve() 将.then函数推入微任务队列(是此次宏任务的小尾巴);
  • 继续执行,遇到 console.log('timeout1') 输出 timeout1;本次宏任务执行完毕;
  • 宏任务执行完毕后紧接着处理小尾巴: .then(res=>{ console.log('这是微任务'); }) 输出 这是微任务;
  • 微任务队列清空后,继续开启下一个宏任务,第二轮tick结束;
  • 将任务队列中的 setTimeout(() => console.log('timeout2'), 10); 回调推入执行栈中执行,输出 timeout2 ; 无微任务,第三轮tick结束,任务队列也为空。

好了,相信经过这两个例子,小伙伴们对事件循环有了初步的认识。接下来我们再顽皮一下:对上面这个demo做一丢丢微调

微调一 : 其他地方不变,then里塞定时器

setTimeout(function() {
  new Promise(resolve=>{
        console.log('000');
      resolve()
    }).then(res=>{
         setTimeout(()=>{
         console.log('这次的执行顺序呢?') -----> 如果这里再塞个定时器呢?执行顺序是什么?
        },10)
        console.log('这是微任务');
    })
  console.log('timeout1');
}, 10);

微调二:其他地方不变,对Promise进行链式调用

new Promise(resolve => {
    console.log('promise1');       
    resolve();
    setTimeout(() => console.log('timeout2'), 10);
}).then(function() {
    console.log('then1')
}).then(()=>{
    console.log('then2')
}).then(()=>{
    console.log('then3')
})

此Promise进行链式调用,其他地方不动,此时的执行顺序是什么?

提示:在一次tick结束时,此tick内微任务队列中的微任务一定会执行完并清空,如果在执行过程中又产生了微任务,那么同样会在此tick过程中执行完毕;而宏任务的执行则可以看成是下一次tick的开始。

3.掺杂async/await

在进行demo解析之前,我们需要补充一下async/await的相关知识点。

async

async相当于隐式返回Promise:当我们在函数前使用async的时候,使得该函数返回的是一个Promise对象,async的函数会在这里帮我们隐式使用Promise.resolve();

下面看个小demo来理解下async函数是怎么隐式转换的:

async function test() {
    console.log('这是async函数')
    return '测试隐式转换' 
}

上面这个async就相当于如下代码:

function test(){
    return new Promise(function(resolve) {
      console.log('这是async函数')
       resolve('测试隐式转换')
   })
}

await

await表示等待,是右侧表达式的结果,这个表达式的计算结果可以是 Promise 对象的值或者一个函数的值(换句话说,就是没有特殊限定)。并且await只能在带有async的内部使用;使用await时,会从右往左执行,当遇到await时,会阻塞函数内部处于它后面的代码,去执行该函数外部的 代码 当外部代码执行完毕,再回到该函数内部执行await后面剩余的代码

好了,补充完前置知识我们来做个demo助助兴:

掺杂async/await的事件循环

async function async2() {				
    console.log('async2');     
}

async function async1() {
    console.log('async1 start');
    await async2();
    console.log('async1 end');
}

console.log('script start');

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

async1();

new Promise((resolve) => {
    resolve()
    console.log('promise1');
}).then(function () {
    console.log('promise2');
});

console.log('script end');

输出顺序为

script start --> async1 start --> async2 --> promise1 --> script end --> async1 end --> promise2 --> setTimeout  

首先为方便理解我们先将async函数转为return Promise的那种形式:

①:
async function async2() {				
    console.log('async2');     
}
转换后如下:
function async2() {			
    return  new Promise(resolve=>{
       console.log('async2');     
    })
}


②:
async function async1() {
    console.log('async1 start');
    await async2();
    console.log('async1 end');
}
转换后如下:
function async1() {
  return new Promise(resolve=>{
    console.log('async1 start');
    #执行async2,并且会阻塞其后面的代码
    console.log('async1 end');
  })
}

所以,最后我们包含async函数的代码块就相当于如下代码:

function async2() {			
    return  new Promise(resolve=>{
       console.log('async2');     
    })
}

function async1() {
  return new Promise(resolve=>{
    console.log('async1 start');
    #执行async2,并且会阻塞其后面的代码,在此处是阻塞了console.log('async1 end')的执行
    console.log('async1 end');
  })
}
=============上面为声明部分===========
console.log('script start');

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

new Promise(resolve=>{
    console.log('async1 start');
    #执行async2,并且会阻塞其后面的代码,在此处是阻塞了console.log(async1end)的执行;这里相当于awaitasync2()
   
    console.log('async1 end');
  })
}

new Promise((resolve) => {
    resolve()
    console.log('promise1');
}).then(function () {
    console.log('promise2');
});

console.log('script end');

经过一系列骚操作之后,我们终于可以来分析这个代码块的执行顺序了,废话不多说,开冲。

解释上述代码:

  • 首先整体代码作为第一个宏任务进入主线程执行栈;
  • 首先顺序执行,遇到了async2、async1 函数的声明,不进行任何输出;
  • 执行到console.log('script start') 输出 script start ;
  • 继续执行,遇到setTimeout宏任务,挂起并推入任务队列;
  • 接着执行Promise内容部分,遇到console.log('async1 start'),输出async1 start ;
  • 这一步重点来了,遇到了await,这该怎么办呢?别急,咱们再来看看使用await会发生什么:使用await时,会从右往左执行,当遇到await时,会阻塞函数内部处于它后面的代码,去执行该函数外部的 代码 当外部代码执行完毕,再回到该函数内部执行await后面剩余的代码
  • 好了,下面开始解释await async2():由于是是从右往左执行,所以我们首先执行了async2()输出了一个Promise,我们执行了Promise的内容输出了async2;async2执行完了之后,遇到await,完全不出意外,后面的代码被阻塞;我们去执行外面的代码;
  • 因为console.log('async1 end')被await阻塞掉了,我们先执行外面的代码:执行了外面Promise的内容,遇到了resolve(),将.then函数推入微任务队列;然后执行console.log('promise1'),输出 promise1;
  • 最后执行到console.log('script end'),输出 script end;
  • 到此,我们外层的代码就执行完毕,现在想想好像少了什么?往前一看,我们console.log('async1 end')还在等待,此时,JS引擎执行log输出 async1 end 。
  • 由此,我们本次的宏任务就执行完毕,下面看看是否有微任务,JS引擎去微任务队列一看,好家伙,还藏着一个 then(function () {console.log('promise2');}); 把此任务回调推到执行栈中执行,输出 promise2;
  • 此次tick执行结束,开启下一个宏任务;
  • 从任务队列拿setTimeout这个宏任务,塞入执行栈执行,打印输出setTimeout,本次无微任务,结束tick;
  • 循环结束;

到此这篇关于实例详解JS中的事件循环机制的文章就介绍到这了,更多相关JS事件循环机制内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 实例详解JS中的事件循环机制

本文链接: https://www.lsjlt.com/news/146738.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 实例详解JS中的事件循环机制
    目录一、前言二、宏、微任务三、Tick 执行顺序四、案例详解1.掺杂setTimeout2.掺杂微任务,此处主要是Promise.then3.掺杂async/await一、前言 之前...
    99+
    2024-04-02
  • 一文详解JS中的事件循环机制
    目录前言1、JavaScript是单线程的2、同步和异步3、事件循环前言 我们知道JavaScript 是单线程的编程语言,只能同一时间内做一件事,按顺序来处理事件,但是在遇到异步事...
    99+
    2024-04-02
  • JS的事件循环执行机制详解
    目录前言JS语言的特点JS中同步和异步的使用事件循环是什么?事件循环执行过程微任务和宏任务的区别JS执行/运行机制最后前言 在前端开发中,涉及到JS原生的使用原理是非常重要的知识点,...
    99+
    2023-05-19
    JS事件循环执行机制 JS事件循环 JS事件
  • JS中事件循环机制的示例分析
    小编给大家分享一下JS中事件循环机制的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1、JavaScript是单线程的JavaScript 是一种单线程的...
    99+
    2023-06-29
  • node.js事件循环机制及与js区别详解
    目录一、是什么二、流程三、题目一、是什么 在浏览器事件循环(opens new window)中,我们了解到javascript在浏览器中的事件循环机制,其是根据HTML5定义的规范...
    99+
    2024-04-02
  • JS中浏览器事件循环机制的示例分析
    这篇文章将为大家详细讲解有关JS中浏览器事件循环机制的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。先来明白些概念性内容。进程、线程进程是系统分配的独立资源,是 ...
    99+
    2024-04-02
  • 关于js的事件循环机制剖析
    前言 众所周知, JavaScript是单线程这一核心,可是浏览器又能很好的处理异步请求,那么到底是为什么呢?其中的原理与事件循环机制大有关系。 在探索事件循环之前,我们得先了解浏览...
    99+
    2024-04-02
  • nodejs中事件循环机制的示例分析
    这篇文章主要介绍了nodejs中事件循环机制的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。前端开发离不开JavaScript,Javascript是一种web前端语...
    99+
    2023-06-14
  • Node.js事件循环机制实例代码分析
    这篇文章主要介绍“Node.js事件循环机制实例代码分析”,在日常操作中,相信很多人在Node.js事件循环机制实例代码分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Node.js事件循环机制实例代码分析...
    99+
    2023-07-04
  • Javascript前端事件循环机制详细讲解
    目录一、消息队列和事件循环1.单线程处理机制2.事件循环机制3.消息队列4.IO线程5.页面使用单线程的缺点二、setTimeout1.浏览器怎么实现 setTimeout2.使用s...
    99+
    2022-12-30
    JavaScript事件循环机制 JS循环机制
  • vue.js的事件循环机制如何理解
    这篇文章主要介绍了vue.js的事件循环机制如何理解的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇vue.js的事件循环机制如何理解文章都会有所收获,下面我们一起来看看吧。一、事件循环机制介绍  &n...
    99+
    2023-06-29
  • 深入了解Javascript的事件循环机制
    目录单线程的Javascript同步 vs 异步 宏任务 vs 微任务定时器To Be Continued单线程的Javascript JavaScript是一种单线程语言,它主要用...
    99+
    2024-04-02
  • Node.js中事件循环的机制是什么
    本篇内容介绍了“Node.js中事件循环的机制是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!先看一个demo:setTimeout((...
    99+
    2023-06-17
  • 详解JavaScript事件循环
    目录一、事件循环的执行过程二、事件循环进阶用法三、JavaScript任务类型3.1 同步任务&异步任务3.2 宏任务&微任务JavaScript事件循环是一种机制,...
    99+
    2023-05-16
    JavaScript事件循环 JavaScript循环
  • JS中的事件冒泡机制实例分析
    这篇文章主要介绍“JS中的事件冒泡机制实例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“JS中的事件冒泡机制实例分析”文章能帮助大家解决问题。1. 事件在浏览器...
    99+
    2024-04-02
  • 深入了解Node事件循环(EventLoop)机制
    主线程从"任务队列"中读取事件,这个过程是循环不断的,所以整个的这种运行机制又称为Event Loop(事件循环)。下面本篇文章就来带大家掌握Node.js中的eventloop,希望对大家有所帮助!虽然js可以在浏览器...
    99+
    2023-05-14
    javascript Node.js 面试 前端
  • Node.js 事件循环的原理与机制
    ...
    99+
    2024-04-02
  • Node事件循环机制是什么
    这篇文章主要介绍“Node事件循环机制是什么”,在日常操作中,相信很多人在Node事件循环机制是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Node事件循环机制是什么”的疑惑有所帮助!接下来,请跟着小编...
    99+
    2023-07-05
  • Node中的事件循环、process.nextTick()实例分析
    这篇文章主要讲解了“Node中的事件循环、process.nextTick()实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Node中的事件循环、p...
    99+
    2024-04-02
  • JS面试之对事件循环的理解
    目录一、是什么事件循环(Event Loop)二、宏任务与微任务微任务宏任务三、async与awaitasyncawait四、流程分析 一、是什么 JavaScript 在...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作