Promise面试题

Promise面试题,带你搞懂同步异步执行顺序

  • 前置知识
  • 面试题
    • 面试题一
    • 面试题二
    • 面试题三
    • 面试题四
  • 分析
    • 面试题一分析
    • 面试题二分析
    • 面试题三分析
    • 面试题四分析



在这里插入图片描述

前置知识

Promise中的then方法
then:指定用于得到成功value的成功回调和用于得到失败reason的失败回调,并且将放回一个新的Promise实例化对象
成功的状态:执行then方法的第一个回调函数 失败的状态:执行then方法的第二个回调函数

then方法的返回值是Promise实例化对象,其状态取决于回调函数的内容

  • 如果返回的为非Promise实例化对象,则得到一个成功的Promise
  • 如果返回的是Promise实例化对象,则Promise实例化对象的状态和结果值将直接影响result常量的状态和结果值
  • 如果为抛出异常,则新的Promise实例化对象(result)为失败的Promise

JS中用来存储待执行回调函数的队列中包含了两个特定的队列
宏队列:用来保存执行的宏任务(回调),比如:定时器、DOM时间操作,ajax操作
微队列:用来保存执行的微任务(回调),比如:promise(then catch finally)
JS执行的时候会区分两个队列

  1. 首先JS引擎会先将所有的同步代码都执行完
  2. 接着在每次准备取出第一个宏任务的执行之前,都需要将所有的微任务一个一个取出来执行
  • 顺序为:同步任务<–微任务<–宏任务

在这里插入图片描述

事件循环的执行顺序可以概括为以下几个步骤(以下答案来自ChatGPT):

  1. 执行同步任务(同步代码),直到执行栈为空。
  2. 执行微任务队列中的所有微任务。微任务包括Promise的回调函数、MutationObserver的回调函数等。
  3. 更新渲染(如果需要)。
  4. 执行下一个宏任务。
    在上述步骤中,微任务队列的执行在渲染之前,也在下一个宏任务之前。这意味着微任务具有更高的优先级,可以在下一个宏任务之前立即执行。

面试题

面试题一

etTimeout(()=>{
	console.log(1);
 },0)
 Promise.resolve().then(()=>{
     console.log(2);
 })
 Promise.resolve().then(()=>{
     console.log(3);
 })
 console.log(4);

//答案:... 

面试题二

setTimeout(()=>{
    console.log(1);
})
new Promise(resolve=>{
    console.log(2);
    resolve()
}).then(()=>{
    console.log(3);
}).then(()=>{
    console.log(4);
})
console.log(5);
//答案:...

面试题三

const first = () => (new Promise((resolve,reject)=>{
    console.log(1);
    let p = new Promise((resolve,reject)=>{
        console.log(2);
        setTimeout(()=>{
            console.log(3);
            resolve(4)
        },0)
        resolve(5)
    })
    resolve(6)
    p.then(arg=>{
        console.log(arg);
    })
}))
first().then(arg=>{
    console.log(arg);
})
console.log(7);
//答案:...

面试题四

在这里插入图片描述

setTimeout(()=>{
     console.log(0);
 },0)
 new Promise((resolve,reject)=>{
     console.log(1);
     resolve();
 }).then(()=>{  //片段一
     console.log(2);
     new Promise((resolve,reject)=>{
         console.log(3);
         resolve();
     }).then(()=>{
         console.log(4);
     }).then(()=>{
         console.log(5);
     })
 }).then(()=>{  // 片段二
     console.log(6);
 })
 new Promise((resolve,reject)=>{
     console.log(7);
     resolve();
 }).then(()=>{  // 片段三
     console.log(8);
 })
 //答案:...

分析

面试题一分析

打印输出:4 2 3 1
分析:定时器为宏任务,加入宏任务队列;then方法为异步任务且为微任务,加入微任务队列;console.log打印输出为同步任务,直接打印输出4。由于微任务队列执行先于宏任务队列,因此先考虑2和3的打印输出,此处由入微任务队列先后顺序而定,先输出2,然后输出3,最后考虑输出宏任务队列的1,虽然其延迟时间为0。

  • 同步任务:4
  • 微任务:2 3
  • 宏任务:1

面试题二分析

打印输出:2 5 3 4 1

分析:定时器为宏任务,加入宏任务队列;new Promise为同步任务,执行函数体内resolve方法,首先打印输出2,然后更改Promise实例对象状态为成功;then方法为异步任务,且为微任务,加入微任务队列;console.log打印处处为同步任务,直接打印输出5;因此首先,打印输出同步任务2和5,然后考虑微任务队列,两个then方法执行先后由队列中顺序而定,输出3和4,最后考虑宏任务队列,打印输出1。

  • 同步任务:2 5
  • 微任务:3 4
  • 宏任务:1

面试题三分析

打印输出:1 2 7 5 6 3

分析:定义first函数为同步任务,执行其函数体,在new Promise的实例化方法中,首先执行同步输出1;定义p为一个Promise实例对象也为同步任务,执行函数体,首先打印输出2;定时器为宏任务,加入宏任务队列;执行resolve(5),更改p的状态为成功;执行resolve(6),更改外层Promise实例对象状态为成功;then方法为微任务,加入微任务队列;执行打印输出7。因此目前先执行同步任务,打印输出1、2和7,然后考虑微任务列队里执行先后,由于resolve(5)已更改p的状态为成功,因此首先输出arg:5,然后由于resolve(6)已更改外层Promise实例对象状态为成功,因此输出arg:6,因此目前执行微任务队列,打印输出5和6,最后考虑宏任务队列,打印输出3,但注意此时不会执行resolve(4)再去更改p的状态,因此状态已经由pending变为 resolved。

  • 同步任务:1 2 7
  • 微任务:5 6
  • 宏任务:3

面试题四分析

打印输出:1 7 2 3 8 4 6 5 0

分析:定时器为宏任务,加入宏任务队列;执行new Promise同步任务,首先打印输出1,接着执行resolve更改其实例化对象的状态为成功;then方法为微任务,加入微任务队列;继续执行new Promise同步任务,首先打印输出7,接着执行resolve更改其实例化对象状态为成功;then方法为微任务,加入微任务队列。因此目前首先执行同步任务,输出1和7,然后考虑微任务队列里执行先后顺序,由于已更改第一个Promise实例对象状态为成功,因此执行其链式调用then方法的第一个回调函数,首先打印输出2,接着执行new Promise函数体,首先打印输出3(微任务里),接着执行resolve,更改其实例化对象的状态为成功。由于then方法是异步的,当片段一(代码中已标出)代码没有执行完,片段二是不会执行的,转而会执行片段三,由于执行片段三前已执行resolve,其实例对象状态已变为成功,因此执行片段三的then方法中第一个回调,输出8,接着回到片段一里的第一个then方法,由于已经执行resolve,其实例对象状态已变为成功,因此执行then方法,输出4,此时若片段一中第一个then没有执行完,也不会接着链式执行第二个then。因此接着执行片段二,由于片段一的then方法没有return,因此执行完then的返回值为undefined,当返回值为非Promise对象:undefined时,则得到一个成功的Promise,且其[[PromiseResult]]为undefined,则可以执行then的第一个回调,输出6,同理回到片段一里的第二个then,输出5,最后执行宏队列任务,输出0

重难点:3之后输出8,4之后输出6,是因为then方法是异步的,前一个没有执行完,会接着往下走

  • 同步任务:1 7
  • 微任务:2 3 8 4 6 5
  • 宏任务:0

then方法是异步执行,就是当【.then()】前的方法执行完后再执行【then()】内部的程序
这样就避免了数据没获取到等的问题,语法为【promise.then(onCompleted,onRejected)】。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/22130.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C语言深度解析--数组

目录 一维数组的创建与初始化 一维数组的创建&#xff1a; 一维数组的初始化&#xff1a; 一维数组的使用&#xff1a; 一维数组在内存中的存储&#xff1a; 二维数组的创建与初始化 二维数组的创建&#xff1a; 二维数组的初始化&#xff1a; 二维数组的使用&#xf…

字节跳动开源其云原生数据仓库 ByConity

动手点关注 干货不迷路 ‍ ‍项目简介 ByConity 是字节跳动开源的云原生数据仓库&#xff0c;它采用计算-存储分离的架构&#xff0c;支持多个关键功能特性&#xff0c;如计算存储分离、弹性扩缩容、租户资源隔离和数据读写的强一致性等。通过利用主流的 OLAP 引擎优化&#xf…

Hive---拉链表设计与实现

1 数据同步问题 Hive在实际工作中主要用于构建离线数据仓库&#xff0c;定期的从各种数据源中同步采集数据到Hive中&#xff0c;经过分层转换提供数据应用。比如每天需要从MySQL中同步最新的订单信息、用户信息、店铺信息等到数据仓库中&#xff0c;进行订单分析、用户分析。 …

使用Gradle7.6.1 + SpringBoot3.0.2 + java17创建微服务项目(学习)

这是一个大胆的决定 这里是导航 技术栈开发工具一、创建gradle父子项目&#xff08;deity&#xff09;1.0 简单流程示意1.1、IDEA中主要图示1.1.1 项目结构图1.1.2 IDEA中 Gradle配置 1.2、deity父项目build.gradle文件1.3、deity父项目settings.gradle文件1.4、子项目build.g…

django ORM框架 第四章 聚合函数

上一章&#xff1a;django ORM框架 第三章 关联表的数据创建与查询_做测试的喵酱的博客-CSDN博客 一、聚合函数类型&#xff1a; from django.db.models import Q, Count, Avg, Max, Min 班级表&#xff1a; 学生信息表&#xff1a; 1.1 Count 模版&#xff1a; QuerySet.…

3 手工推导Neural Networ

线性模型假设的问题 如上图&#xff0c;对非线性类边界的数据进行分类 一个解决方案是将数据映射到更高维的空间&#xff0c;就变成线性可分的了。 ϕ \phi ϕ 是一个映射函数&#xff0c;将x从一个低维空间映射到高维空间。 ϕ \phi ϕ 可不可以是一个线性函数&#xff1f; …

音视频源码调试前准备vs2019+qt5.15.2搭建可调试环境

安装vs2019qt,并且在windows环境上安装ffmpeg&#xff0c;尝试使用qtcdb进行调试&#xff0c;尝试使用vs2019加载qt的程序。 安装VS20195.12.2qt环境&#xff0c;并进行测试。 1&#xff1a;安装Visual Studio 2019, a.从官网下载&#xff0c;或者vs2019社区版本下载地址 ht…

Cloud Studio 内核升级之触手可及

前言 Cloud Studio是基于浏览器的集成式开发环境&#xff08;IDE&#xff09;&#xff0c;为开发者提供了一个永不间断的云端工作站。用户在使用 Cloud Studio 时无需安装&#xff0c;随时随地打开浏览器就能使用。云端开发体验与本地几乎一样&#xff0c;上手门槛更低&#x…

前端axios fetch 解决接口请求响应数据返回快慢不均导致的数据错误问题

引言 搜索功能&#xff0c;我想很多业务都会涉及&#xff0c;这个功能的特点是&#xff1a; 用户可以在输入框中输入一个关键字&#xff0c;然后在一个列表中显示该关键字对应的数据&#xff1b;输入框是可以随时修改/删除全部或部分关键字的&#xff1b;如果是实时搜索&…

查看电脑的BIOS版本的五种方法

查看主板BIOS版本的五种方法 概述1. 在 BIOS 中查看2. 使用 DirectX 诊断工具3. 使用 CPU-Z 中查看4. 在 CMD 中查看&#xff08;一&#xff09;5. 在 CMD 中查看 &#xff08;二&#xff09;结束语 概述 BIOS是 Basic Input Output System 的缩略词&#xff0c;直译就是 **基…

【数据结构与算法】- 周测四

课程链接: 清华大学驭风计划 代码仓库&#xff1a;Victor94-king/MachineLearning: MachineLearning basic introduction (github.com) 驭风计划是由清华大学老师教授的&#xff0c;其分为四门课&#xff0c;包括: 机器学习(张敏教授) &#xff0c; 深度学习(胡晓林教授), 计算…

InnoDB数据页结构

什么是页&#xff1f;什么是数据页&#xff1f; 页是InnoDB管理存储空间的基本单元&#xff0c;一个页的大小一般是16k。 InnoDB有许多不同的页&#xff0c;有存放表空间头部信息的页&#xff0c;INODE信息的页&#xff0c;当然还有存放我们记录信息的页&#xff0c;这个页叫…

车载以太网 - SomeIP - 协议用例 - Messages_01

目录 Service Discovery Messages 1、验证Instance ID为0xFFFF时DUT需要返回该Serveice ID包含的所有Instance ID

「实验记录」MIT 6.824 Raft Lab2B Log Replication

#Lab2B - Log Replication I. SourceII. My CodeIII. MotivationIV. SolutionS1 - leader上任即初始化S2 - leader发送AppendEntriesS3 - follower接收AppendEntriesS4 - leader收到AppendEntries 回信S5 - candidate选举限制S6 - defs.go约定俗成和实现Start() V. Result I. S…

LeetCode 栈和队列OJ题目分享

目录 有效的括号&#xff08;括号匹配&#xff09;用栈实现队列用队列实现栈设计循环队列 有效的括号&#xff08;括号匹配&#xff09; 链接: link 题目描述&#xff1a; 题目思路&#xff1a; 1、如果是左括号“&#xff08; { [ ”就入栈 2、如果是右括号“&#xff09; }…

程序员:面试造火箭,入职拧螺丝?太难了···

刚开始工作的时候&#xff0c;我也想不通这个问题&#xff0c;甚至很鄙视这种现象。后面当了面试官&#xff0c;做到了公司中层管理&#xff0c;也会站在公司以及行业角度去重新思考这个问题。 为什么这种现象会越来越普遍呢&#xff1f;尤其在 IT 行业愈加明显。 面试看的是…

Packet Tracer – VLAN 实施故障排除场景 2

Packet Tracer – VLAN 实施故障排除场景 2 拓扑图 地址分配表 设备 接口 IPv4 地址 子网掩码 默认网关 S1 VLAN 56 192.168.56.11 255.255.255.0 不适用 S2 VLAN 56 192.168.56.12 255.255.255.0 不适用 S3 VLAN 56 192.168.56.13 255.255.255.0 不适用 P…

头歌计算机组成原理实验—运算器设计(11)第11关:MIPS运算器设计

第11关&#xff1a;MIPS运算器设计 实验目的 学生理解算术逻辑运算单元&#xff08;ALU&#xff09;的基本构成&#xff0c;掌握 Logisim 中各种运算组件的使用方法&#xff0c;熟悉多路选择器的使用&#xff0c;能利用前述实验完成的32位加法器、 Logisim 中的运算组件构造指…

图神经网络:(处理点云)PointNet++的实现

文章说明&#xff1a; 1)参考资料&#xff1a;PYG官方文档。超链。 2)博主水平不高&#xff0c;如有错误还望批评指正。 3)我在百度网盘上传了这篇文章的jupyter notebook和有关文献。超链。提取码8848。 文章目录 简单前置工作学习文献阅读PointNet的实现模型问题 简单前置工作…

智慧井盖监测终端,智能井盖-以科技解决智慧城市“顽疾”,守护城市生命线

平升电子智慧井盖监测终端,智能井盖-以科技解决智慧城市“顽疾”,守护城市生命线-智慧井盖&#xff0c;实现对井下设备和井盖状态的监测及预警&#xff0c;是各类智慧管网管理系统中不可或缺的重要设备&#xff0c;解决了井下监测环境潮湿易水淹、电力供应困难、通讯不畅等难题…