JavaScript重难点突破:期约与异步函数

同步和异步

  1. 同步(Synchronous)​
  • 定义:任务按顺序依次执行,前一个任务完成前,后续任务必须等待。

  • 特点:阻塞性执行,程序逻辑直观,但效率较低

  1. 异步(Asynchronous)​
  • 定义:任务发起后无需等待结果,程序继续执行其他操作,待任务完成后通过回调或事件通知处理结果。

  • 特点:非阻塞性执行,支持并发,资源利用率高

简单来说,同步就是刷牙然后煮面,异步就是让面在一边煮着一边跑去刷牙。

期约

期约是对尚不存在结果的一个替身。

在ES6中,期约是一种引用类型(Promise),使用new操作符实例化。

期约状态机

期约对象有三种状态,这些状态是期约对象内置的,除了调用相应的API,否则不能对其进行更改。

  • pending(待定)

  • fulfilled(兑现)

  • rejected(拒绝)

新建一个期约对象,并且期约对象还没进行任何操作时,期约对象的状态为pending,当期约对象已经被成功解决后,则转为fulfilled,而解决失败则转为rejected,具体让期约状态转换的函数后面介绍。

待定(pending)是期约的最初始状态。在待定状态下,期约可以落定(settled)为代表成功的兑现(fulfilled)状态,或者代表失败的拒绝(rejected)状态。
无论落定为哪种状态都是不可逆的。只要从待定转换为兑现或拒绝,期约的状态就不再改变。而且,也不能保证期约必然会脱离待定状态。因此,组织合理的代码无论期约解决(resolve)
还是拒绝(reject),甚至永远处于待定(pending)状态,都应该具有恰当的行为。
重要的是,期约的状态是私有的,不能直接通过JavaScript检测到。这主要是为了避免根据读取到的期约状态,以同步方式处理期约对象。
另外,期约的状态也不能被外部JavaScript代码修改。这与不能读取该状态的原因一样:【期约故意将异步行为封装起来,从而隔离外部的同步代码】

《JavaScript高级程序设计第四版》

如何理解期约对象

正如前面介绍的,期约对象是专门为异步编程而设计的,期约对象是对尚不存在结果的一个替身。

举个例子,你参加了一场考试,试卷由他人进行批改(异步操作),而你在试卷批改的过程中是自由的,你可以吃饭睡觉,你也可以尝试查询试卷批改的状态。

如果试卷还在批改当中,则返回pending,如果已经批改结束,并且你已经通过了考试,返回fulfilled,如果未通过考试,则返回rejected

理解了上述的场景,我们就能够理解期约对象和期约对象的状态机了。

  • 期约对象代表了某一个异步操作。

  • 期约状态机代表了异步操作的完成状态。

如何控制期约对象的状态机

期约对象的状态是私有的,只能通过期约对象内部的API进行操作。

向期约对象中传入一个执行器函数(回调函数),期约对象会向执行器函数传入两个参数resolvereject,调用resolve()使期约状态变为fulfilled,调用reject()使期约状态变为rejected

    const waiter1 = new Promise((resolve, reject) => { });const waiter2 = new Promise((resolve, reject) => resolve());const waiter3 = new Promise((resolve, reject) => reject());// 其中undefined表示期约对象在完成后期待一个返回值,但这里没有给返回值console.log('waiter1:', waiter1); // waiter1: Promise {<pending>}console.log('waiter2:', waiter2); // waiter2: Promise {<fulfilled>:undefined}console.log('waiter3:', waiter3); // waiter1: Promise {<rejected>:undefined}

请添加图片描述

期约对象的静态方法

  • Promise.resolve()

    这个方法可以直接创建一个fulfilled状态的期约对象,这个期约对象的值由传入的参数指定。

        const settled1 = Promise.resolve(3)const settled2 = Promise.resolve('我是字符串')const settled3 = Promise.resolve(new Promise(() => {}))console.log(settled1) // Promise {<fulfilled>: 3}console.log(settled2) // Promise {<fulfilled>: '我是字符串'}console.log(settled3) // Promise {<pending>}
    

    可以看到,我们传入什么值,期约对象就会返回什么值。

    但是如果我们传入的是另一个期约对象,则会直接返回传入的期约对象。

  • Promise.reject()

    这个方法可以直接创建一个rejected状态的期约对象,这个期约对象的值由传入的参数指定。

        const settled1 = Promise.reject(3)const settled2 = Promise.reject('我是字符串')const settled3 = Promise.reject(new Promise(() => {}))console.log(settled1) // Promise {<rejected>: 3}console.log(settled2) // Promise {<rejected>: '我是字符串'}console.log(settled3) // Promise {<rejected>: Promise {<pending>}}
    

    这个方法和Promise.resolve()类似,也会将传入的值作为期约对象的值返回。

    但不同的是,如果传入一个期约对象,那么这个期约对象也会作为期约对象的值返回。

    调用reject()或者Promise.reject()都会抛出一个异步错误。同步代码块中的trycatch结构无法捕获到异步错误,只有异步结构中才能捕获异步错误

    期约的实例方法

  • 实现Thenable方法

    在ECMAScript暴露的异步结构中,任何对象都有一个then()方法。

  • Promise.prototype.then()

    then()方法挂载在Promise的原型上,所以被所有Promise实例对象共享。

    then()方法接收两个回调函数参数,第一个参数在Promise对象的状态落定为fulfilled时执行,第二个参数在Promise对象的状态落定为rejected时执行。

    如何理解then()方法

    在平时写js方法时,我们都是使用的同步代码块,也就是说,写在后面的代码一定后执行,比如我们在前面一行计算let sum = 10 + 1,那么我们就可以在这一行后面的任意位置输出sum,因为sum的计算是写在前面的,在同步代码块中,他已经被计算完毕了。

    现在我们使用了异步编程,我们已经知道Promise对象内置了一个状态机,它用于通知自己是否执行完毕。

    由于Promise是异步执行的,假如我们在同步代码块中读取Promise对象,我们有可能获取3种结果。如果我们需要打印Promise的返回值,我们不可能在同步代码块中不停的检测Promise的状态,这样会导致后面的代码无法执行,这就违背了异步编程的初衷。

    所以我们使用then()方法,then()方法可以想象为一个触发器,设置好then()方法后,只要Promise对象settled到了任意一种状态,就会触发then()方法中设定好的函数,这样我们就可以异步的处理Promise的返回值而无需再在同步代码块中处理Promise的返回值了。

        const p1 = new Promise((resolve, reject) => {// 1秒后返回10 + 1的计算结果setTimeout(() => resolve(10 + 1), 1000);});p1.then(() => {console.log('我计算完成,被触发了');console.log('我是then中的p1:',p1)}, () => { console.log('我计算失败,被触发了'); });console.log('我是同步代码块中的p1:',p1)
    

    请添加图片描述

    可以看到,同步代码块由于执行的比较快,已经运行到输出Promise对象的值了,但是此时Promise对象还没执行完,状态为pending,而then中却可以正常输出Promise的返回值11,这是因为then()中的第一个参数只有在Promise对象状态为fulfilled时才被调用。

    我们将then()方法的第一个参数称为onResolved处理程序,第二个参数称为onRejected处理程序。

  • .then() 返回的 Promise 状态如何确定?

    回调返回值类型决定状态

    • 返回普通值​(非 Promise 对象):新 Promise 会被 Promise.resolve() 包装为 ​fulfilled 状态

      p.then(() => 42); // 新 Promise 状态:fulfilled,值:42
      
    • 抛出异常:新 Promise 变为 ​rejected 状态,异常对象作为拒绝原因

      p.then(() => { throw new Error("fail"); }); // 状态:rejected,原因:Error对象
      
    • 返回 Promise 对象:新 Promise 将 ​继承该 Promise 的状态和值

      p.then(() => Promise.reject("error")); // 新 Promise 状态:rejected,原因:"error"
      

    因此通过.then()方法返回的也是Promise对象,所以也有.then()方法,.then()方法可以进行链式调用

        const p1 = new Promise((resolve, reject) => {// 3秒后返回10 + 1的计算结果setTimeout(() => resolve(double(1)), 1000);});p1.then(value => {return value}).then(value => {return double(value)}).then(value => {return double(value)}).then(value => console.log(value)) // 8
    
  • Promise.prototype.catch()

    等于then(null,() => {}),也就是onRejected处理程序。

  • Promise.prototype.finally()

    传入finally()的回调函数能保证一定被执行,和try-catch-finally中的finally用法一致。

  • Promise.all()和Promise.race()

    这两个方法都可以传入一个包含多个期约的可迭代对象,常见方法是传入一个包含多个期约的数组。

    • Promise.all():会等待传入的期约全部兑现后才兑现,如果有一个期约待定或者拒绝,则返回待定或者拒绝

    • Promise.race():会返回根据第一个兑现或者拒绝的期约决定状态的新期约对象。

异步函数

通过刚刚期约对象的学习我们了解到,期约对象是异步执行的,因此如果想操作期约对象的流程必须要使用then()方法。

但是这样同样导致了一个问题,同步代码块和异步代码块被完全的区分开了,从使用了期约对象开始,所有和这个期约对象有关的流程都要在then中实现,这会使得then()方法的函数体变得很大,并不好维护。

于是,从ES8开始引入了一组新的关键字async/await用于解决这个问题。

基本概念

  • async 函数

    • 声明方式:在函数前添加 async 关键字,如 async function fetchData() {}

    • 返回值:始终返回一个 Promise 对象。若函数返回非 Promise 值(如字符串、数值),该值会被自动包装为 resolve 状态的 Promise

      async function example() { return "Hello"; }
      example().then(console.log); // 输出 "Hello"
      
  • await 关键字

    • 使用范围:仅能在 async 函数内部使用。

    • 功能:暂停当前 async 函数的执行,等待右侧的 Promise 完成(resolvereject),并返回解析后的值

           async function fetchUser() {const response = await fetch('/api/user'); // 等待请求完成return response.json();
      }
      

    简单来说,关键字async声明了这个函数应该被异步执行,关键字await表明被async声明的函数应该被停止执行,等到await右侧的表达式返回值后才被继续执行。

async

async关键字标记的函数会返回一个Promise对象,如果返回的值不是Promise对象,则会使用Promise.resolve()对返回的值进行包装。

async 函数的执行流程

  1. 同步代码的立即执行
  • 未遇到 awaitasync 函数内部的代码会按照同步顺序立即执行,与普通函数的行为完全一致。例如:

    async function demo() {console.log("A");  // 同步执行console.log("B");  // 同步执行
    }
    demo();
    console.log("C");
    

    输出顺序为:A → B → C

  • 本质async 函数被调用时,其函数体内的同步代码会直接进入主线程的同步任务队列,立即执行。

  1. ​**await 对执行流程的干预**
  • 遇到 await:函数会暂停当前执行,将 await 后的表达式(通常是 Promise)放入微任务队列,并交出主线程控制权。此时,外部同步代码会继续执行。例如:

    async function demo() {console.log("A");await new Promise(resolve => setTimeout(resolve, 1000)); // 暂停console.log("B");  // 异步执行(微任务)
    }
    demo();
    console.log("C");
    

    输出顺序为:A → C → (1秒后) B

  • 关键机制await 后的代码会被封装为微任务,等待当前同步代码执行完毕后才会继续执行。

    async function heavyTask () {console.log("开始耗时操作");// 没有awaitfor (let i = 0; i < 1e9; i++);  console.log("耗时操作完成");}heavyTask();console.log("外部代码");

请添加图片描述

    async function heavyTask () {console.log("开始耗时操作");// 有awaitawait '123'for (let i = 0; i < 1e9; i++);  console.log("耗时操作完成");}heavyTask();console.log("外部代码");

请添加图片描述

可以看到,在await后的代码才会作为异步代码执行,否则async修饰的代码会像普通函数一样同步执行。

await

await关键字期待右侧是一个实现了Thenable接口的对象。

但如果不是,则await不会等待,而是将右侧视为一个已经fulfilled的期约对象,直接返回。不会将值包装为Promise对象

    const func = async () => {console.log(await '123') }// 注意,123没有被包装为Promise对象func() // '123'

如果await右侧是一个Promise对象,并且尚未settled,那么异步程序会在await处阻塞,停止运行直到右侧的Promise对象已经fulfilled或者rejected

异步函数的特质不会扩展到嵌套函数,await只能在async标记的函数中使用

如果需要进行并行优化,不要每调用一次async函数就等待await返回值,而是一次性将async函数全部调用后,再按照需要的顺序等待await的返回值。

    const asyncFunc1 = async () => { console.log(1); return 'a' }const asyncFunc2 = async () => { console.log(2); return 'b' }const asyncFunc3 = async () => { console.log(3); return 'c' }const asyncFunc4 = async () => { console.log(4); return 'd' }// 错误示范const run1 = async () => {console.log(await asyncFunc1()) console.log(await asyncFunc2()) console.log(await asyncFunc3()) console.log(await asyncFunc4()) }// 正确示范const run2 = async () => {const res1 = asyncFunc1()const res2 = asyncFunc2()const res3 = asyncFunc3()const res4 = asyncFunc4()console.log(await res1) console.log(await res2) console.log(await res3) console.log(await res4) }

在错误示范中,每次调用async函数都等到async函数返回后才执行下一个async函数。
而正确示范中,将所有async函数全部执行后,再等待async函数的返回值。

正确示范也可以通过Promise.all()来实现。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

学习总结 网格划分+瞬态求解设置

网格划分部分 1.导入几何文件 导入我们的几何模型&#xff0c;他的格式为.scdocx 2.添加局部尺寸BOI 因为要对对前缘和尾缘进行局部加密&#xff0c;所以进行一个BOI的局部加密&#xff0c;目标尺寸取的几何尺寸的最小尺寸的0.1&#xff0c;就是0.4mm。 3.生成表面网格 表面…

.NET 使用 WMQ 连接Queue 发送 message 实例

1. 首先得下载客户端&#xff0c;没有客户端无法发送message. 安装好之后长这样 我装的是7.5 安装目录如下 tools/dotnet 目录中有演示的demo 2. .Net 连接MQ必须引用bin目录中的 amqmdnet.dll 因为他是创建Queuemanager 的核心库&#xff0c; 项目中引用using IBM.WMQ; 才…

风电行业预测性维护解决方案:给风机装上 “智能医生”,实现故障 “秒级预警”

引言&#xff1a;风电设备故障为何成为 “运维黑洞”&#xff1f; 某海上风电场因齿轮箱轴承故障停机 3 天&#xff0c;直接损失 50 万元发电量。传统维护模式下&#xff0c;人工巡检覆盖率不足 40%&#xff0c;故障修复平均耗时 72 小时。而预测性维护通过物联网 AI 技术&am…

5、无线通信基站的FPGA实现架构

基站&#xff08;Base Station&#xff0c;BS&#xff09;&#xff0c;也称为公用移动通信基站&#xff0c;是无线电台站的一种形式&#xff0c;具体则指在一定的无线电覆盖区中&#xff0c;通过移动通信交换中心&#xff0c;与移动电话终端之间的信息传递的无线电收发信电台。…

笔记2——网络参考模型

一、OSI参考模型&#xff1a; 应用层&#xff1a; 报文 给应用程序提供接口 表示层&#xff1a; 进行数据格式的转换 会话层&#xff1a; 在通讯双方之间建立、管理和终止会话 传输层&#xff1a; 数据段&#xff1b;建立、维护、取消一次端到端的数据传输过程&#xff1b;控制…

最短路径:Bellman-Ford算法

Bellman-Ford的操作步骤 1.初始化距离&#xff1a;将起点的dist值设置为0&#xff0c;其他点的dist值设置为无穷大。 2.执行n-1轮松弛操作&#xff1a;遍历所有边&#xff0c;更新最短距离&#xff0c;收敛后可获得最短路径。 3.检测负权环&#xff1a;额外遍历一次&#xf…

0402-对象和类(访问器 更改器 日期类)

OOP&#xff1a;面向对象程序设计 类&#xff1a;构造对象的模板或蓝图 类构造对象的过程称为创建类的实例 封装&#xff1a;对外隐藏数据的真实实现方式&#xff0c;提供简单的方法 &#xff08;类比方向盘&#xff09; 对象&#xff1a;本质上是内存中的一小块空间 识别类&a…

【 <二> 丹方改良:Spring 时代的 JavaWeb】之 Spring Boot 中的文件上传与下载:实现文件管理功能

<前文回顾> 点击此处查看 合集 https://blog.csdn.net/foyodesigner/category_12907601.html?fromshareblogcolumn&sharetypeblogcolumn&sharerId12907601&sharereferPC&sharesourceFoyoDesigner&sharefromfrom_link <今日更新> 一、开篇整…

搜索算法------DFS练习2

1. 题目 2. 思路和题解 从题目中可以看出&#xff0c;如果一个格子上有雨水&#xff0c;那么就可以流到周围比他高度低的单元格&#xff0c;如果单元格和海洋相邻&#xff0c;那么雨水也会流入海洋。总而言之一句话就是水从高处流向低处。从这里的流向可以联想到深度优先搜索这…

[python] 正则表达式

1.分割str s"1-2--3---4" are.findall(r\d|[-],s) # 输出&#xff1a;[1, -, 2, --, 3, ---, 4]s"-4(2(3)" # ? 表示 - 可以出现0次或1次 # \d 表示匹配一个或多个连续数字 # \D 表示匹配非数字字符 sre.findall(r-?\d|\D,s) # 输出&#xff1a;[-4, (,…

定制化管理系统与通用管理系统,谁更胜一筹?

一、定制化管理系统与通用管理系统的定义与特点 定制化管理系统 定制化管理系统是根据企业的具体业务需求和流程进行个性化开发的软件系统。它能够深度贴合企业的管理需求&#xff0c;提供高度灵活的解决方案。其特点包括&#xff1a; 高度适应性&#xff1a;能够精准匹配企业…

gitee 配置git上传

Git入门&#xff1f;查看 帮助 , Visual Studio / TortoiseGit / Eclipse / Xcode 下如何连接本站, 如何导入仓库 简易的命令行入门教程: Git 全局设置: 以 176fuguM2项目为例 git config --global user.name "堕落圣甲虫" git config --global user.email "11…

SpringBoot+Vue 中 WebSocket 的使用

WebSocket 是一种在单个 TCP 连接上进行全双工通信的协议&#xff0c;它使得客户端和服务器之间可以进行实时数据传输&#xff0c;打破了传统 HTTP 协议请求 - 响应模式的限制。 下面我会展示在 SpringBoot Vue 中&#xff0c;使用WebSocket进行前后端通信。 后端 1、引入 j…

STM32 FATFS - 在SDIO的SD卡中运行fatfs

参考文章 STM32CubeMX | SD Card FATFS - 知乎 [STM32F4]基于F407的硬件移植Free RTOSFATFS&#xff08;SDIO&#xff09;_freertosfatfs-CSDN博客 例程地址&#xff1a;STM32FatFS: 基于stm32的fatfs例程&#xff0c;配合博客文章 基于梁山派天空星开发板&#xff0c;STM3…

Java 进化之路:从 Java 8 到 Java 21 的重要新特性

Java 进化之路&#xff1a;从 Java 8 到 Java 21 的重要新特性 开篇介绍 在软件开发领域&#xff0c;Java 作为一门历史悠久且广泛应用的编程语言&#xff0c;始终保持着其核心竞争力和持续创新能力。自 Java 8 发布以来&#xff0c;Java 经历了一系列重要版本更新&#xff0…

Reactor 事件流 vs. Spring 事件 (ApplicationEvent)

Reactor 事件流 vs. Spring 事件 ApplicationEvent Reactor 事件流 vs. Spring 事件 (ApplicationEvent)1️⃣ 核心区别2️⃣ Spring 事件 (ApplicationEvent)✅ 示例&#xff1a;Spring 事件发布 & 监听1️⃣ 定义事件2️⃣ 发布事件3️⃣ 监听事件&#x1f539; 进阶&…

JVM生产环境问题定位与解决实战(六):总结篇——问题定位思路与工具选择策略

本文已收录于《JVM生产环境问题定位与解决实战》专栏&#xff0c;完整系列见文末目录 引言 在前五篇文章中&#xff0c;我们深入探讨了JVM生产环境问题定位与解决的实战技巧&#xff0c;从基础的jps、jmap、jstat、jstack、jcmd等工具&#xff0c;到JConsole、VisualVM、MAT的…

【5090d】配置运行和微调大模型所需基础环境【一】

RuntimeError: Failed to import transformers.integrations.bitsandbytes because of the following error (look up to see its traceback): No module named triton.ops 原因&#xff1a;是因为在导入 transformers.integrations.bitsandbytes 时缺少必要的依赖项 triton.op…

华为交换综合实验——VRRP、MSTP、Eth-trunk、NAT、DHCP等技术应用

一、实验拓扑 二、实验需求 1,内网Ip地址使用172.16.0.0/16分配 2,sw1和SW2之间互为备份 3, VRRP/STP/VLAN/Eth-trunk均使用 4,所有Pc均通过DHCP获取IP地址 5,ISP只能配置IP地址 6,所有电脑可以正常访问IsP路由器环回 三、需求分析 1、设备连接需求 二层交换机&#xff08;LS…

DeepSeek 开源的 3FS 如何?

DeepSeek 3FS&#xff08;Fire-Flyer File System&#xff09;是一款由深度求索&#xff08;DeepSeek&#xff09;于2025年2月28日开源的高性能并行文件系统&#xff0c;专为人工智能训练和推理任务设计。以下从多个维度详细解析其核心特性、技术架构、应用场景及行业影响&…