我的编程空间,编程开发者的网络收藏夹
学习永远不晚

JS中promise的回调和setTimeout的回调哪个先执行

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

JS中promise的回调和setTimeout的回调哪个先执行

这篇文章将为大家详细讲解有关JS中promise的回调和setTimeout的回调哪个先执行,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。

首先提一个小问题:运行下面这段 JS 代码后控制台的输出是什么?

console.log("script start");setTimeout(function () {  console.log("setTimeout1");}, 0);new Promise((resolve, reject) => {  setTimeout(function () {    console.log("setTimeout2");    resolve();  }, 100);}).then(function () {  console.log("promise1");});Promise.resolve()  .then(function () {    console.log("promise2");  })  .then(function () {    console.log("promise3");  });console.log("script end");

可以先尝试自己分析一下结果,然后再看答案:

script start
script end
promise2
promise3
setTimeout1
setTimeout2
promise1

怎么样,你猜对了吗?如果对这个输出结果感到很迷惑,这篇文章或许可以帮到你。

PS:文中按照标准分析理论结果,但实际上各个浏览器对任务队列的支持情况很混乱,所以如果你在浏览器执行代码后发现结果不同也不必纠结;总体来说 Chrome 的支持比较好。

如果对 Promise 的用法还不熟悉,可以看我的上一篇博客:前端 | JS Promise:axios 请求结果后面的 .then() 是什么意思?

任务 VS 微任务

JavaScript 设计的本质是单线程语言,但随着硬件性能的飞速发展,纯单线程已经不太能够满足需求了。因此 JS 逐渐发展出了任务和微任务,来模拟实现多线程。

浏览器中,对于每个网页(有时也可能是多个同源网页),网页的代码和浏览器自身的用户界面程序运共享同一个主线程,它除了运行浏览器交给它的 JS 代码,也负责收集和派发事件、渲染和绘制网页内容等等。因此,如果主线程中的某个任务阻塞了,其他任务都会受到影响;这就是为什么有时候网页代码出现了错误会导致整个网页渲染失败。

每个主线程都由一个事件循环 Event loops 驱动。事件循环可以理解为一个任务队列,JS 引擎不断的进行“循环-等待”,按顺序处理队列中的任务。事件循环中的任务称作“任务 Task”,由宿主环境(浏览器)创建;每个任务都是宿主计划执行的 JavaScript 代码,如程序初始化、解析HTML、事件触发的回调(例如点击网页上的按钮),或是由 setTimeout() setInterval() 等 API 添加的回调函数。

JS 引擎在执行一个任务的过程中,有时会进行一些异步操作,不会立即执行,但又想在同一个任务中完成、不留到事件循环中的下一个任务里;例如常用的 promise、监控 DOM 的回调等。这时,JS 引擎会创建一个“微任务 Mircotask”,并加入当前的微任务队列中。(有时为了区分,也把任务task称为“宏任务”。)

事件循环、任务、微任务的示意图如下:

JS中promise的回调和setTimeout的回调哪个先执行

执行过程

一个主线程的执行过程如下:

  • 拿出事件循环中的下一个任务

  • 执行任务本身的 Script 代码;期间可能会往任务队列、微任务队列创建添加新任务

  • script 执行完后,检查微任务队列

    • 如果有微任务,顺序执行,期间可能还会创建新的任务和微任务

    • 如果微任务队列为空,这个任务执行结束,回到第一步

可以看出,在一个任务中会反复检查微任务队列,直到没有微任务存在了才会执行下一个任务。因此在任务脚本和微任务脚本中创建的所有微任务都会在这个任务结束前执行,同时也意味着会早于其他所有创建的任务执行(因为新建的任务都加入了任务队列)。

JS中promise的回调和setTimeout的回调哪个先执行

案例分析

明白了任务和微任务的区别,下面再来看文章开头的例子:

console.log("script start");setTimeout(function () {  console.log("setTimeout1");}, 0);new Promise((resolve, reject) => {  setTimeout(function () {    console.log("setTimeout2");    resolve();  }, 100);}).then(function () {  console.log("promise1");});Promise.resolve()  .then(function () {    console.log("promise2");  })  .then(function () {    console.log("promise3");  });console.log("script end");

接下来逐步跟踪代码的执行过程;如果感觉文字不够直观,可以看这篇博客中给出的逐步执行动画。

整个 Script 会被宿主环境传给 JS 引擎,作为任务队列中的一个任务;首先执行任务中的脚本代码:

  • line1: console.log("script start") 是同步代码,直接输出

  • line3: 执行 setTimeout(),在0秒后将 console.log("setTimeout1"); 加入任务队列

  • line8: 执行 setTimeout(),在0.1秒后将 console.log("setTimeout2"); 和 resolve() 加入任务队列

  • line16: 返回一个已成功的 promise,第一个 then 回调被加入微任务队列

  • line24: console.log("script end") 是同步代码,直接输出

  • 任务 script 执行完毕

此时:

  • 控制台输出了 script start script end

  • 任务队列中(除当前任务以外)有2个任务(两个 setTimeout() 的回调按时间先后顺序排列)

  • 微任务队列中有1个任务(promise 的回调)

接下来检查微任务队列,执行队首的微任务:

  • console.log("promise2") 输出

  • 隐式 return,相当于返回一个 Promise.resolve(undefined);因此 Promise 链中的下一个 then 回调被加入微任务队列

  • 微任务执行完毕

此时:

  • 控制台输出了 script start script end promise2

  • 任务队列中(除当前任务以外)有2个任务(两个 setTimeout() 的回调按时间先后顺序排列)

  • 微任务队列中有1个任务(第二个 promise 回调)

再次检查微任务队列,执行队首的微任务:

  • console.log("promise3") 输出

  • 隐式 return(但此时 Promise 链已经结束了,所以无事发生)

  • 微任务执行完毕

此时:

  • 控制台输出了 script start script end promise2 promise3

  • 任务队列中(除当前任务以外)有2个任务(两个 setTimeout() 的回调按时间先后顺序排列)

  • 微任务队列为空

检查微任务队列,发现没有微任务了,当前任务结束;开始执行任务队列中的下一个任务(0秒后执行的回调):

  • console.log("setTimeout1"); 输出

  • 任务 script 执行完毕

此时:

  • 控制台输出了 script start script end promise2 promise3 setTimeout1

  • 任务队列中(除当前任务以外)有1个任务

  • 微任务队列为空

检查微任务队列,发现没有微任务,当前任务结束;开始执行任务队列中的下一个任务(0.1秒后执行的回调):

  • console.log("setTimeout2"); 输出

  • resolve(); 将 promise 的状态更改为已成功;then 回调被加入微任务队列

  • 任务 script 执行完毕

此时:

  • 控制台输出了 script start script end promise2 promise3 setTimeout1 setTimeout2

  • 任务队列中只有当前任务

  • 微任务队列中有一个任务(promise 的回调)

检查微任务队列,执行队首的微任务:

  • console.log("promise1") 输出

  • 隐式 return(但此时 Promise 链已经结束了,所以无事发生)

  • 微任务执行完毕

此时:

  • 控制台输出了 script start script end promise2 promise3 setTimeout1 setTimeout2 promise1

  • 任务队列中只有当前任务

  • 微任务队列为空

检查微任务队列,发现没有微任务,当前任务结束。任务队列中没有其他任务,执行完毕。

关于“JS中promise的回调和setTimeout的回调哪个先执行”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

JS中promise的回调和setTimeout的回调哪个先执行

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

下载Word文档

猜你喜欢

JS中promise的回调和setTimeout的回调哪个先执行

这篇文章将为大家详细讲解有关JS中promise的回调和setTimeout的回调哪个先执行,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。首先提一个小问题:运行下面这段 JS 代码后控制台的输出是什么?c
2023-06-29

在goroutine中执行的IO回调

来到编程网的大家,相信都是编程学习爱好者,希望在这里学习Golang相关编程知识。下面本篇文章就来带大家聊聊《在goroutine中执行的IO回调》,介绍一下,希望对大家的知识积累有所帮助,助力实战开发!问题内容我是 golang 的初学者
在goroutine中执行的IO回调
2024-04-04

介绍JS异步的执行原理和回调方法

本篇内容介绍了“介绍JS异步的执行原理和回调方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、JS异步的执行原理  我们知道JavaSc
2023-06-07

js中可选的回调函数有哪些

这篇文章主要为大家展示了“js中可选的回调函数有哪些”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“js中可选的回调函数有哪些”这篇文章吧。1、then的回调函数不是严格要求必写的。如果不写,pr
2023-06-25

如何执行类型为受多个返回类型约束的泛型类型的回调?

问题内容我试图在我的函数中允许一个允许多个返回元组的回调参数。为了实现这一点,我使用泛型来定义回调的参数类型。func get[in any,out any,translatefn func(in) out | func(in) (ou
如何执行类型为受多个返回类型约束的泛型类型的回调?
2024-02-12

PHP中的array_walk()函数:如何对数组中的每个元素执行回调函数

PHP中的array_walk()函数:如何对数组中的每个元素执行回调函数在PHP中,数组是一种常用的数据结构,经常需要对数组中的每个元素进行特定操作。array_walk()函数可以帮助我们实现这个目标。本文将介绍array_walk()
PHP中的array_walk()函数:如何对数组中的每个元素执行回调函数
2023-11-03

使用PHP的array_map()函数对数组中的每个元素执行回调函数

使用PHP的array_map()函数对数组中的每个元素执行回调函数PHP是一种广泛应用于Web开发的脚本语言,拥有丰富的内置函数和方法。其中,array_map()函数是一个非常有用的函数,可以用于对数组中的每个元素执行回调函数,并返回处
使用PHP的array_map()函数对数组中的每个元素执行回调函数
2023-11-04

使用PHP的array_walk_recursive()函数对多维数组中的每个元素执行回调函数

PHP中的array_walk_recursive()函数可以对多维数组中的每个元素进行遍历,并调用指定的回调函数对其进行操作。本文将通过具体的代码示例来演示如何使用该函数。首先,让我们创建一个多维数组,用于演示array_walk_rec
使用PHP的array_walk_recursive()函数对多维数组中的每个元素执行回调函数
2023-11-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录