当前位置: 首页 > news >正文

【JS】Promise详解

概述

在 JavaScript 中,Promise 是一个表示异步操作最终完成或失败的对象。它本质上是一个返回的对象,你可以附加回调函数,而不是将回调传递给函数。

let promise = new Promise((resolve, reject) => {let condition = true; // 这可以是某个操作的结果// 1秒后检查条件并解决或拒绝 PromisesetTimeout(() => {if (condition) {resolve('Promise 已完成!');} else {reject('Promise 被拒绝!');}}, 1000);
});// 将 then() 和 catch() 处理程序附加到 Promise
promise.then(value => {// 如果 Promise 已解决,则执行此操作console.log(value); // 输出: Promise 已完成!}).catch(error => {// 如果 Promise 被拒绝,则执行此操作console.log(error);});

在这个示例中,创建了一个 Promise,它将在 1 秒后解决或拒绝,具体取决于条件的值。如果 Promise 成功,将调用 resolve 函数;如果 Promise 失败,将调用 reject 函数。

当 Promise 解决时,then 方法被调用,并接收传递给 resolve 函数的值。类似地,当 Promise 被拒绝时,catch 方法被调用,并接收传递给 reject 函数的值。

setTimeout 和事件循环

setTimeout 函数在这个问题中发挥了关键作用。它是一个方法,调用一个函数或在指定的毫秒数后评估一个表达式。在 JavaScript 中,setTimeout 用于延迟代码的执行。

console.log("启动定时器...");setTimeout(() => {console.log("定时已完成!");
}, 2000);

在这个示例中,“启动定时器…”将立即记录到控制台。然后,调用 setTimeout 函数,传递两个参数:一个回调函数和以毫秒为单位的延迟。回调函数是一个简单的箭头函数,用于将 “定时已完成!”记录到控制台,延迟为 2000 毫秒(或2秒)。

一旦调用了 setTimeout,JavaScript 运行时设置了定时器,但随后立即继续执行任何后续代码。它不会暂停或等待定时器完成,这展示了 JavaScript 的非阻塞特性。

在指定的延迟之后(在本例中为2秒),将回调函数添加到任务队列中。但重要的是要注意,回调函数不一定会在此刻立即执行。回调函数实际执行之前的实际延迟可能会比指定的延迟稍长。这是由于事件驱动的 JavaScript 运行时和单线程事件循环的性质所决定的。

假设主 JavaScript 线程中有一个耗时较长的进程或操作。在这种情况下,即使定时器在后台完成,回调函数仍必须等待阻塞任务的完成。这是因为事件循环一次只能处理一个任务,并按照排队的顺序处理任务。

因此,setTimeout 中指定的“2 秒”应被理解为在调用回调函数之前的 “最小延迟”,而不是 “保证延迟”。如果 JavaScript 运行时正忙于其他任务,回调函数实际执行的时间可能会超过 2 秒。这种行为强调了理解 JavaScript 异步性质的重要性,因为它对代码的性能和行为产生重大影响。

另外,值得一提的是 clearTimeout,这是 JavaScript 定时器函数套件中的一个有用函数。clearTimeout 是一个函数,它取消了先前通过调用 setTimeout 建立的定时器。

下面是它的使用示例:

console.log("启动定时器...");// setTimeout 返回一个 Timeout 对象,可用于引用定时器
let timeoutId = setTimeout(() => {console.log("定时已完成!");
}, 2000);// 一些条件或逻辑
if (/* 一些条件 */) {// 取消定时器clearTimeout(timeoutId);
}

如果 if 语句内部的条件为真,那么 clearTimeout 函数将取消通过 setTimeout 设置的定时器。如果取消了定时器,setTimeout 提供的函数将不会被调用。

这在各种场景中很有用,例如,如果要在操作执行之前检查用户是否仍然活跃在页面上,但用户在延迟结束之前导航到其他页面,你可以使用 clearTimeout 来取消检查。

JavaScript 的事件循环

JavaScript 使用调用堆栈来管理函数的执行。当调用函数时,它会被添加到堆栈中。当函数完成时,它会从堆栈中移除。由于 JavaScript 是单线程的,一次只能执行一个函数。

然而,如果一个函数需要较长时间才能执行(例如网络请求),这可能会有问题。这就是事件循环的用武之地。

事件循环 是一个持续的循环,检查调用堆栈是否为空。如果为空,它会从任务队列(也称为事件队列或回调队列)中获取第一个任务并将其推送到调用堆栈中,立即执行它。

异步回调

setTimeout 是 JavaScript 中的异步函数示例。当调用 setTimeout 函数时,它会启动一个定时器,然后立即返回,允许 JavaScript 运行时在等待定时器完成的同时继续执行其他代码。这是 JavaScript 的非阻塞特性。

一旦定时器完成,给 setTimeout 提供的回调函数将添加到任务队列中。事件循环不断检查调用堆栈和任务队列。当调用堆栈为空时,它会从任务队列中获取第一个任务并将其推送到调用堆栈中以执行。

并发和事件循环

以下是 JavaScript 如何处理并发操作的方式:

  1. JavaScript 运行一段代码(此代码在主线程上运行)。
  2. 当遇到异步操作(如 setTimeout、fetch 等)时,JavaScript 将其启动,然后继续运行其余代码。它不会等待异步操作完成。这些异步操作可能在后台运行,但不会运行在主 JavaScript 线程上。
  3. 当异步操作完成时,其回调函数将被放入任务队列中。
  4. 一旦调用堆栈为空(即,当前事件循环的所有代码都已执行完毕),事件循环将从任务队列中获取第一个任务并将其推送到调用堆栈中,以立即执行它。

这个过程继续进行,事件循环会在调用堆栈为空时将任务从任务队列中推送到调用堆栈,从而使 JavaScript 能够处理多个操作,尽管它是单线程的。

这是 JavaScript 处理异步操作的高级概述。实际上更复杂,还涉及微任务和宏任务等附加特性,但这是其基本概念。查看事件循环详解。

Async/await

Async/await 可以被看作是 Promise 的语法糖,它使异步代码更易于编写和理解。当我们使用 async 关键字标记函数时,它将成为一个自动返回 promise 的异步函数。在异步函数中,我们可以使用 await 关键字暂停代码的执行,直到 promise 解决或拒绝。

通过使用 await,我们可以消除使用 promises 时通常需要的明确的 .then() 和 .catch() 链。相反,我们可以以更线性和类似同步代码的方式构建代码。这使得更容易理解程序的流程并以更简洁的方式处理错误。

示例:

// 使用 promises 和显式的 .then() 和 .catch()
fetchData().then(response => {// 处理响应console.log("响应:", response);return processData(response);}).then(result => {// 处理处理后的数据console.log("处理后的数据:", result);}).catch(error => {// 处理任何错误console.error("错误:", error);});// 使用 async/await
async function fetchDataAndProcess() {try {const response = await fetchData();console.log("响应:", response);const result = await processData(response);console.log("处理后的数据:", result);} catch (error) {console.error("错误:", error);}
}
fetchDataAndProcess();

通过使用明确的 .then() 和 .catch() 链,我们必须分别处理异步操作的每个步骤。当涉及多个 promises 时,可能会变得复杂,从而导致嵌套或链接的 .then() 调用。此外,错误处理需要单独的 .catch() 块。

相比之下,第二个示例使用 async/await 在更线性和类似同步代码的方式中构建代码。fetchDataAndProcess() 函数标记为 async,允许我们在其中使用 await 关键字。这消除了明确的 .then() 和 .catch() 链的需要。

在底层,await 关键字会暂停函数的执行,使其他任务继续执行,比如处理用户输入或动画。JavaScript 引擎会切换到执行其他代码,直到由异步函数返回的 promise 解决,然后它将恢复异步函数中的其余代码的执行。

Promise 链式编程

Promise 链式编程是 JavaScript 中的一种技术,允许你按顺序执行多个异步操作,每个操作在前一个操作完成后启动。Promise 链的主要优点是,它允许你避免使用嵌套回调来处理异步代码时的 “回调地狱” 或 “回调金字塔”。相反,可以编写几乎看起来像同步代码的异步代码,从而更容易理解和维护。Promise 链中的每个 then 方法都会接收上一个 promise 解决的结果。此结果可以用来通知链中的下一个步骤。如果链中的 promise 被拒绝,后续的 then 方法将被跳过,直到找到 catch 方法来处理错误。

fetchData().then(response => {console.log("响应:", response);return processData(response);  // 这返回一个新的 promise}).then(processedData => {console.log("处理后的数据:", processedData);return furtherProcessing(processedData);  // 这返回另一个新的 promise}).then(finalResult => {console.log("最终结果:", finalResult);}).catch(error => {console.error("错误:", error);});

fetchDataprocessDatafurtherProcessing 都是异步函数,返回 promises。then 方法被链接在一起,每个 then 在前一个 promise 解决后开始其操作。如果链中的任何 promise 被拒绝,将调用最后的 catch 方法来处理错误。

理解 .finally

在 JavaScript 中,Promises 提供了处理异步操作及其结果的几种强大方法之一就是 .finally 方法。.finally 方法是 Promise 的内置方法,它始终会执行,无论 promise 是否被解决。这使得它成为放置无论 promise 结果如何都必须运行的清理代码的绝佳位置。

let isLoading = true;fetch('https://api.example.com/data').then(response => {if (!response.ok) {throw new Error('网络响应不正常');}return response.json();}).then(data => console.log(data)).catch(error => console.error('错误:', error)).finally(() => {isLoading = false;console.log('获取操作完成');});

使用 fetch(返回一个 promise)从 URL 获取数据。然后,我们使用 .then 处理响应,并使用 .catch 处理任何错误。最后,无论获取操作是否成功,都会调用 .finally 以将 isLoading 设置为 false 并在控制台上记录一条消息。

理解异步函数中的 promise 返回

在解决这个问题时,可能会派上用场的一个有趣事实是,在异步函数中,无论你是返回 return new Promise() 还是 return await new Promise(),行为通常是相同的。这是因为异步函数始终将返回值包装在 promise 中。然而,在某些情况下,如错误处理,使用 await 可能会有所不同。

思考下面的示例:

async function example() {try {return new Promise((resolve, reject) => {throw new Error('糟糕!');});} catch (err) {console.error(err);}
}example(); // 错误不会被捕获,它拒绝了 example 返回的 promise。async function example2() {try {return await new Promise((resolve, reject) => {throw new Error('糟糕!');});} catch (err) {console.error(err);}
}example2(); // 错误被捕获,example2 返回的 promise 被解决。

在 example 函数中,try 块不会捕获 promise 引发的错误,因为 promise 在引发错误之前就已经返回了。在 example2 中,await 会导致函数等待 promise 完成或抛出错误,因此它可以捕获 promise 引发的错误。

相关文章:

【JS】Promise详解

概述 在 JavaScript 中,Promise 是一个表示异步操作最终完成或失败的对象。它本质上是一个返回的对象,你可以附加回调函数,而不是将回调传递给函数。 let promise new Promise((resolve, reject) > {let condition true; // 这可以是某…...

原生微信小程序如何动态配置主题颜色及如何调用子组件的方法

一、最终效果 二、步骤 1、在初始化进入项目时,获取当前主题色 2、把主题色定义成全局变量(即在app.js中设置) 3、tabBar也需要定义全局变量,在首页时需要重新赋值 三、具体实现 1、app.js onLaunch () {//获取主题数据this.set…...

Java关键字(1)

Java中的关键字是指被编程语言保留用于特定用途的单词。这些关键字不能用作变量名或标识符。以下是Java中的一些关键字: public:表示公共的,可以被任何类访问。 private:表示私有的,只能被定义该关键字的类访问。 cl…...

【机器学习合集】深度生成模型 ->(个人学习记录笔记)

深度生成模型 深度生成模型基础 1. 监督学习与无监督学习 1.1 监督学习 定义 在真值标签Y的指导下,学习一个映射函数F,使得F(X)Y 判别模型 Discriminative Model,即判别式模型,又称为条件模型,或条件概率模型 生…...

Java将PDF转换为文本

在Java中&#xff0c;你可以使用现有的库来将PDF文件转换为文本。下面是一个简单的示例&#xff0c;使用Apache PDFBox库来实现PDF到文本的转换。首先&#xff0c;确保在你的项目中添加了Apache PDFBox库的依赖。你可以在 Maven 项目中添加以下依赖&#xff1a; <!--Pdf--&g…...

Linux 运维工具之1Panel

一、1Panel 简介 1Panel 是一个现代化、开源的 Linux 服务器运维管理面板。 特点&#xff1a; 快速建站&#xff1a;深度集成 Wordpress 和 Halo&#xff0c;域名绑定、SSL 证书配置等一键搞定&#xff1b;高效管理&#xff1a;通过 Web 端轻松管理 Linux 服务器&#xff0…...

深入了解小红书笔记详情API:为内容创新提供动力

一、小红书笔记详情API简介 小红书笔记详情API是一种允许开发者访问小红书平台上的笔记详细数据的接口。通过这个API&#xff0c;我们可以获取笔记的标题、内容、标签、点赞数、评论数等详细信息。这些数据对于内容创作者和品牌来说至关重要&#xff0c;可以帮助他们了解用户喜…...

Animate 2024(Adobe an2024)

Animate 2024是一款由Adobe公司开发的动画和互动内容创作工具&#xff0c;是Flash的演进版本。Animate 2024为设计师和开发者提供了更丰富的功能&#xff0c;让他们能够创建各种类型的动画、交互式内容和多媒体应用程序。 Animate 2024具有以下特点&#xff1a; 强大的设计工…...

尽量避免删改List

作者简介&#xff1a;大家好&#xff0c;我是smart哥&#xff0c;前中兴通讯、美团架构师&#xff0c;现某互联网公司CTO 联系qq&#xff1a;184480602&#xff0c;加我进群&#xff0c;大家一起学习&#xff0c;一起进步&#xff0c;一起对抗互联网寒冬 学习必须往深处挖&…...

【Linux操作系统】探秘Linux奥秘:用户、组、密码及权限管理的解密与实战

&#x1f308;个人主页&#xff1a;Sarapines Programmer&#x1f525; 系列专栏&#xff1a;《操作系统实验室》&#x1f516;诗赋清音&#xff1a;柳垂轻絮拂人衣&#xff0c;心随风舞梦飞。 山川湖海皆可涉&#xff0c;勇者征途逐星辉。 目录 &#x1fa90;1 初识Linux OS &…...

计算机组成原理复习4

习题 练习题 下列不属于系统总线的为&#xff08;&#xff09; a.数据总线 b.地址总线 c.控制总线 d.片内总线 D 系统总线中地址总线的功能是&#xff08;&#xff09; a.选择主存单元地址 b.选择进行信息传输的设备 c.选择外存地址 d.指定主存和I/O设备接口电路的地址 D 解…...

AutoSAR(基础入门篇)3.3-Autosar中RTE的数据一致性与Interface接口

目录 一、RTE的数据一致性 1、什么是数据一致性 2、数据一致性的实现机制 2.1、利用RTE管理<...

超维空间S2无人机使用说明书——52、初级版——使用PID算法进行基于yolo的目标跟踪

引言&#xff1a;在实际工程项目中&#xff0c;为了提高系统的响应速度和稳定性&#xff0c;往往需要采用一定的控制算法进行目标跟踪。这里抛砖引玉&#xff0c;仅采用简单的PID算法进行目标的跟随控制&#xff0c;目标的识别依然采用yolo。对系统要求更高的&#xff0c;可以对…...

<JavaEE> TCP 的通信机制(一) -- 确认应答 和 超时重传

目录 TCP的通信机制的核心特性 一、确认应答 1&#xff09;什么是确认应答&#xff1f; 2&#xff09;如何“确认”&#xff1f; 3&#xff09;如何“应答”&#xff1f; 二、超时重传 1&#xff09;丢包的概念 2&#xff09;什么是超时重传&#xff1f; 3&#xff09…...

Spark任务调度与数据本地性

Apache Spark是一个分布式计算框架&#xff0c;用于处理大规模数据。了解Spark任务调度与数据本地性是构建高效分布式应用程序的关键。本文将深入探讨Spark任务调度的流程、数据本地性的重要性&#xff0c;并提供丰富的示例代码来帮助大家更好地理解这些概念。 Spark任务调度的…...

【论文阅读】Self-Paced Curriculum Learning

论文下载 代码 Supplementary Materials bib: INPROCEEDINGS{,title {Self-Paced Curriculum Learning},author {Lu Jiang and Deyu Meng and Qian Zhao and Shiguang Shan and Alexander Hauptmann},booktitle {AAAI},year {2015},pages {2694--2700} }1. 摘…...

C++简易线程池

原理说明&#xff1a; 1. 线程池创建时&#xff0c;指定线程池的大小thread_size。当有新的函数任务通过函数addFunction ()添加进来后&#xff0c;其中一个线程执行函数。一个线程一次执行一个函数。如果函数数量大与线程池数量&#xff0c;则后来的函数等待。 2. 线程池内部…...

【MATLAB】PSO粒子群优化LSTM(PSO_LSTM)的时间序列预测

有意向获取代码&#xff0c;请转文末观看代码获取方式~也可转原文链接获取~ 1 基本定义 PSO粒子群优化LSTM&#xff08;PSO-LSTM&#xff09;是一种将粒子群优化算法&#xff08;PSO&#xff09;与长短期记忆神经网络&#xff08;LSTM&#xff09;相结合的混合模型。该算法通过…...

产品经理学习-怎么写PRD文档

目录 瀑布流方法论介绍 产品需求文档&#xff08;PRD&#xff09;介绍 产品需求文档的基本要素 撰写产品需求文档 优先产品需求文档的特点 其他相关文档 瀑布流方法论介绍 瀑布流模型是一种项目的开发和管理的方法论&#xff0c;是敏捷的开发管理方式相对应的另一种方法…...

第3课 获取并播放音频流

本课对应源文件下载链接&#xff1a; https://download.csdn.net/download/XiBuQiuChong/88680079 FFmpeg作为一套庞大的音视频处理开源工具&#xff0c;其源码有太多值得研究的地方。但对于大多数初学者而言&#xff0c;如何快速利用相关的API写出自己想要的东西才是迫切需要…...

Spark编程实验四:Spark Streaming编程

目录 一、目的与要求 二、实验内容 三、实验步骤 1、利用Spark Streaming对三种类型的基本数据源的数据进行处理 2、利用Spark Streaming对Kafka高级数据源的数据进行处理 3、完成DStream的两种有状态转换操作 4、把DStream的数据输出保存到文本文件或MySQL数据库中 四…...

Flink去重计数统计用户数

1.数据 订单表&#xff0c;分别是店铺id、用户id和支付金额 "店铺id,用户id,支付金额", "shop-1,user-1,1", "shop-1,user-2,1", "shop-1,user-2,1", "shop-1,user-3,1", "shop-1,user-3,1", "shop-1,user…...

力扣:62. 不同路径(动态规划,附python二维数组的定义)

题目&#xff1a; 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish” &#xff09;。 问总共有多少条不同的路径&…...

2022年全球运维大会(GOPS深圳站)-核心PPT资料下载

一、峰会简介 GOPS 主要面向运维行业的中高端技术人员&#xff0c;包括运维、开发、测试、架构师等群体。目的在于帮助IT技术从业者系统学习了解相关知识体系&#xff0c;让创新技术推动社会进步。您将会看到国内外知名企业的相关技术案例&#xff0c;也能与国内顶尖的技术专家…...

8868体育助力意甲罗马俱乐部 迪巴拉有望付出

8868体育助力意甲罗马俱乐部 迪巴拉有望付出 意甲罗马俱乐部是8868体育合作球队之一&#xff0c;本赛季&#xff0c;在意甲第14轮的比赛中&#xff0c;罗马客场2-1战胜萨索洛&#xff0c;积分上升到意甲第4位。 有报道称&#xff0c;迪巴拉在对阵佛罗伦萨的比赛中受伤&#xff…...

java设计模式实战【策略模式+观察者模式+命令模式+组合模式,混合模式在支付系统中的应用】

引言 在代码开发的世界里&#xff0c;理论知识的重要性毋庸置疑&#xff0c;但实战经验往往才是知识的真正试金石。正所谓&#xff0c;“读万卷书不如行万里路”&#xff0c;理论的学习需要通过实践来验证和深化。设计模式作为软件开发中的重要理论&#xff0c;其真正的价值在…...

小程序wx:if 和hidden的区别?

在小程序中&#xff0c;wx:if 和 hidden 是用于条件渲染的两种不同方式。 选择使用哪种方式取决于具体情况。如果条件变化频繁或节点包含复杂的子节点&#xff0c;可以考虑使用 wx:if 进行条件渲染&#xff1b;如果条件变化较少且节点结构简单&#xff0c;可以使用 hidden 控制…...

自动驾驶学习笔记(二十三)——车辆控制模型

#Apollo开发者# 学习课程的传送门如下&#xff0c;当您也准备学习自动驾驶时&#xff0c;可以和我一同前往&#xff1a; 《自动驾驶新人之旅》免费课程—> 传送门 《Apollo开放平台9.0专项技术公开课》免费报名—>传送门 文章目录 前言 运动学模型 动力学模型 总结…...

Linux Shell 015-文本双向覆盖重定向工具tee

Linux Shell 015-文本双向覆盖重定向工具tee 本节关键字&#xff1a;Linux、Bash Shell、文本双向覆盖重定向工具 相关指令&#xff1a;tee、echo、cat tee介绍 tee工具是从标准输入读取并写入到标准输出和文件&#xff0c;即&#xff1a;双向覆盖重定向&#xff08;屏幕输出…...

【PyQt】(自定义类)QIcon派生,更易用的纯色Icon

嫌Qt自带的icon太丑&#xff0c;自己写了一个&#xff0c;主要用于纯色图标的自由改色。 当然&#xff0c;图标素材得网上找。 Qt原生图标与现代图标对比&#xff1a; 没有对比就没有伤害 Qt图标 网络素材图标 自定义类XJQ_Icon&#xff1a; from PyQt5.QtGui import QIc…...

oa办公系统如何使用/seo运营是什么意思

1. 介绍 1.1 定义 外观模式&#xff1a;提供了一个统一的接口&#xff0c;用来访问子系统中的一群接口。外观定义了一个高层接口&#xff0c;让子系统更容易使用。 1.2 结构图 1.3 角色 外观(Facade)角色 &#xff1a; 客户端可以调用这个角色的方法。此角色知晓相关的&#…...

哪些网站可以做帮助文档/市场调研报告模板范文

1.打开页眉&#xff0c;将光标定位在页眉&#xff0c;打开“边框和底纹”选项卡&#xff0c;在“边框”栏中右下角选择段落&#xff0c;在右上角‘设置’下点击“无&#xff08;N&#xff09;”&#xff0c;确定就OK啦&#xff01;如图所示 2进入页眉编辑&#xff0c;直接将样式…...

广州公司建站模板/seo推广什么意思

最近金融市场像过山车一样&#xff0c;高高低低&#xff0c;哀声一片&#xff0c;媒体称为全球股灾。虽然金融市场极其让人失望&#xff0c;但是互联网垂直B2B市场却异常的火爆&#xff0c;一两年时间&#xff0c;冒出了不少垂直B2B电商平台&#xff0c;最为显眼的&#xff0c;…...

旅游网站建设规划方案/sem是指什么

传送门 首先应该考虑一下&#xff0c;多加一条边对树上路径的影响是什么 可以发现多加一条边就会出现一颗基环树 我们发现所有经过环的路径都会翻倍 那么假设\(x\)是\(y\)的父亲&#xff0c;\(x\)在环内&#xff0c;那么将\(y\)加入环的贡献就是\((size[x]-size[y])*size[y]\) …...

湖南网站建设企业/关键词优化师

const readline require(readline-sync);//引入用户输入功能console.log(请输入数组的长度&#xff1a;);//提示用户输入let length readline.question();//获取用户输入let arr [];//创建数组for (let i 0; i < length; i) {console.log(请输入数组的第${i 1}个数&…...

wordpress文章字号/女生学电子商务好吗

转自&#xff1a;http://bbs.9ria.com/thread-210322-1-1.html 首先&#xff0c;从copy开始说&#xff0c;简而言之&#xff0c;copy的目的就是生成一个新的实例&#xff0c;然后把其成员都按原实例赋值。对于非指针型的成员&#xff0c;比如BOOL, int, float&#xff0c;这样的…...