掌握 JavaScript 异步模式:从回调到异步/等待
知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个文章开发实战,手把手教大家学习《掌握 JavaScript 异步模式:从回调到异步/等待》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!
当我第一次遇到异步 javascript 时,我在回调方面遇到了困难,并且不知道 promises 在幕后是如何工作的。随着时间的推移,对 promise 和 async/await 的了解改变了我的编码方法,使其更易于管理。在本博客中,我们将逐步探索这些异步模式,揭示它们如何简化您的开发流程并使您的代码更干净、更高效。让我们一起深入探讨并揭开这些概念!
为什么需要学习异步 javascript?
学习异步 javascript 对于现代 web 开发至关重要。它允许您有效地处理 api 请求等任务,使您的应用程序保持快速响应。掌握异步技术(例如 promises 和 async/await)不仅对于构建可扩展的应用程序至关重要,而且对于在 javascript 工作面试中取得成功也至关重要,理解这些概念通常是重点。通过掌握异步 javascript,您将提高编码技能并更好地为现实世界的挑战做好准备。
什么是异步模式?
javascript 中的异步模式是用于处理需要时间的任务的技术,例如从服务器获取数据,而不冻结应用程序。最初,开发人员使用回调来管理这些任务,但这种方法通常会导致代码复杂且难以阅读,称为“回调地狱”。为了简化这一点,引入了 promise,通过链接操作和更优雅地处理错误,提供了一种更干净的方式来处理异步操作。 async/await 继续发展,它允许您编写看起来和行为更像同步代码的异步代码,从而更易于阅读和维护。这些模式对于构建高效、响应式的应用程序至关重要,也是现代 javascript 开发的基础。我们将在本博客中更详细地探讨这些概念。
什么是回调?
回调 是作为参数传递给其他函数的函数,目的是让接收函数在某个时刻执行回调。这对于您希望确保某些代码在特定任务完成后运行的场景非常有用,例如从服务器获取数据或完成计算后。
回调如何工作:
- 您定义一个函数(回调)。
- 您将此函数作为参数传递给另一个函数。
- 接收函数在适当的时间执行回调。
示例 1
function fetchdata(callback) { // simulate fetching data with a delay settimeout(() => { const data = "data fetched"; callback(data); // call the callback function with the fetched data }, 1000); } function processdata(data) { console.log("processing:", data); } fetchdata(processdata); // fetchdata will call processdata with the data
示例 2
// function that adds two numbers and uses a callback to return the result function addnumbers(a, b, callback) { const result = a + b; callback(result); // call the callback function with the result } // callback function to handle the result function displayresult(result) { console.log("the result is:", result); } // call addnumbers with the displayresult callback addnumbers(5, 3, displayresult);
注意:我认为回调对于处理异步操作是有效的,但要小心:随着代码复杂性的增加,尤其是嵌套回调,您可能会遇到称为回调地狱的问题。当回调彼此深度嵌套时,就会出现此问题,从而导致可读性问题并使代码难以维护。
回调地狱
回调地狱(也称为末日金字塔)指的是有多个嵌套回调的情况。当您需要按顺序执行多个异步操作,并且每个操作都依赖于前一个操作时,就会发生这种情况。
例如。这会创建一个难以阅读和维护的“金字塔”结构。
fetchdata(function(data1) { processdata1(data1, function(result1) { processdata2(result1, function(result2) { processdata3(result2, function(result3) { console.log("final result:", result3); }); }); }); });
回调地狱问题:
- 可读性:代码变得难以阅读和理解。
- 可维护性:进行更改或调试变得具有挑战性。
- 错误处理:管理错误可能会变得复杂。
使用回调处理错误
使用回调时,通常使用称为错误优先回调的模式。在此模式中,回调函数将错误作为其第一个参数。如果没有错误,第一个参数通常为 null 或未定义,实际结果作为第二个参数提供。
function fetchdata(callback) { settimeout(() => { const error = null; // or `new error("some error occurred")` if there's an error const data = "data fetched"; callback(error, data); // pass error and data to the callback }, 1000); } function processdata(error, data) { if (error) { console.error("error:", error); return; } console.log("processing:", data); } fetchdata(processdata); // `processdata` will handle both error and data
注意:在回调之后,引入了 promise 来处理 javascript 中的异步过程。我们现在将更深入地研究 promise 并探索它们在幕后是如何工作的。
promise 简介
promises 是表示异步操作最终完成(或失败)及其结果值的对象。与回调相比,它们提供了一种更简洁的方式来处理异步代码。
承诺的目的:
- 避免回调地狱: promise 有助于管理多个异步操作,无需深层嵌套。
- 提高可读性: promise 提供了一种更具可读性的方式来处理异步任务序列。
承诺国家
promise 可以处于以下三种状态之一:
- pending: promise 被解决或拒绝之前的初始状态。
- fulfilled: 操作成功完成并且已调用resolve 时的状态。
- rejected: 操作失败时的状态,并且已调用reject。
注意:如果您想探索更多内容,您应该查看了解 promise 如何在幕后工作,其中我讨论了 promise 如何在幕后工作。
示例 1
// creating a new promise const mypromise = new promise((resolve, reject) => { const success = true; // simulate success or failure if (success) { resolve("operation successful!"); // if successful, call resolve } else { reject("operation failed!"); // if failed, call reject } }); // using the promise mypromise .then((message) => { console.log(message); // handle the successful case }) .catch((error) => { console.error(error); // handle the error case });
示例 2
const examplepromise = new promise((resolve, reject) => { settimeout(() => { const success = math.random() > 0.5; // randomly succeed or fail if (success) { resolve("success!"); } else { reject("failure."); } }, 1000); }); console.log("promise state: pending..."); // to check the state, you would use `.then()` or `.catch()` examplepromise .then((message) => { console.log("promise state: fulfilled"); console.log(message); }) .catch((error) => { console.log("promise state: rejected"); console.error(error); });
链接承诺
链接允许您按顺序执行多个异步操作,每一步都取决于前一步的结果。
链式 promise 是 javascript 的一项强大功能,它允许您执行一系列异步操作,其中每一步都取决于前一步的结果。与深层嵌套的回调相比,这种方法更干净、更具可读性。
promise 链如何工作
promise 链 涉及按顺序连接多个 promise。链中的每个 promise 仅在前一个 promise 解决后才会执行,并且每个 promise 的结果都可以传递到链中的下一步。
function step1() { return new promise((resolve) => { settimeout(() => resolve("step 1 completed"), 1000); }); } function step2(message) { return new promise((resolve) => { settimeout(() => resolve(message + " -> step 2 completed"), 1000); }); } function step3(message) { return new promise((resolve) => { settimeout(() => resolve(message + " -> step 3 completed"), 1000); }); } // chaining the promises step1() .then(result => step2(result)) .then(result => step3(result)) .then(finalresult => console.log(finalresult)) .catch(error => console.error("error:", error));
链接的缺点:
虽然与嵌套回调相比,链接 promise 提高了可读性,但如果链变得太长或太复杂,它仍然会变得笨拙。这可能会导致类似于回调地狱中出现的可读性问题。
注意:为了解决这些挑战,引入了 async 和 wait 来提供一种更易读、更简单的方法来处理 javascript 中的异步操作。
异步/等待简介
async 和 await 是 javascript 中引入的关键字,旨在使处理异步代码更具可读性且更易于使用。
- async:将函数标记为异步。异步函数总是返回一个承诺,并且它允许在其中使用等待。
- await:暂停异步函数的执行,直到承诺解决,从而更容易以类似同步的方式处理异步结果。
async function fetchdata() { return new promise((resolve) => { settimeout(() => { resolve("data fetched"); }, 1000); }); } async function getdata() { const data = await fetchdata(); // wait for fetchdata to resolve console.log(data); // logs "data fetched" } getdata();
异步/等待如何工作
1。异步函数始终返回 promise:
无论你从异步函数返回什么,它总是会被包装在一个承诺中。例如:
async function example() { return "hello"; } example().then(console.log); // logs "hello"
即使 example() 返回一个字符串,它也会自动包装在一个 promise 中。
2。等待暂停执行:
await 关键字暂停异步函数的执行,直到它等待的 promise 解析。
async function example() { console.log("start"); const result = await new promise((resolve) => { settimeout(() => { resolve("done"); }, 1000); }); console.log(result); // logs "done" after 1 second } example();
在此示例中:
- “开始”立即被记录。
- await 暂停执行,直到 promise 在 1 秒后解决。
- 承诺解决后会记录“完成”。
使用 async/await 进行错误处理
使用 async/await 处理错误是使用 try/catch 块完成的,这使得错误处理比 promise 链更加直观。
async function fetchdata() { throw new error("something went wrong!"); } async function getdata() { try { const data = await fetchdata(); console.log(data); } catch (error) { console.error("error:", error.message); // logs "error: something went wrong!" } } getdata();
使用 promises,您可以使用 .catch() 处理错误:
fetchdata() .then(data => console.log(data)) .catch(error => console.error("error:", error.message));
将 async/await 与 try/catch 结合使用通常会产生更干净、更易读的代码。
将 async/await 与 promise 结合起来
您可以将 async/await 与现有的基于 promise 的函数无缝结合使用。
示例
function fetchData() { return new Promise((resolve) => { setTimeout(() => { resolve("Data fetched"); }, 1000); }); } async function getData() { const data = await fetchData(); // Wait for the promise to resolve console.log(data); // Logs "Data fetched" } getData();
最佳实践:
- 使用 async/await 来提高可读性:在处理多个异步操作时,async/await 可以让代码更加线性,更容易理解。
- 与 promise 结合:继续使用 async/await 和基于 promise 的函数来更自然地处理复杂的异步流程。
- 错误处理: 始终在异步函数中使用 try/catch 块来处理潜在的错误。
结论
与传统的 promise 链和回调相比,async 和 wait 提供了一种更清晰、更易读的方式来处理异步操作。通过允许您编写外观和行为类似于同步代码的异步代码,它们可以简化复杂的逻辑并使用 try/catch 块改进错误处理。将 async/await 与 promise 一起使用会产生更易于维护和理解的代码。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《掌握 JavaScript 异步模式:从回调到异步/等待》文章吧,也可关注golang学习网公众号了解相关技术文章。

- 上一篇
- 在 Golang 中如何调试匿名函数?

- 下一篇
- Golang 函数如何影响程序性能?
-
- 文章 · 前端 | 2分钟前 | 性能优化 Intl.NumberFormat 数字格式化 千位分隔 百分比格式
- JavaScript数字格式化技巧与实现攻略
- 394浏览 收藏
-
- 文章 · 前端 | 23分钟前 |
- IntersectionObserverAPI在JavaScript中的使用方法
- 135浏览 收藏
-
- 文章 · 前端 | 34分钟前 |
- requestAnimationFrame在JavaScript中怎么用?
- 150浏览 收藏
-
- 文章 · 前端 | 37分钟前 | JavaScript join split 字符串反转 reverse
- JavaScript字符串反转技巧与实现
- 217浏览 收藏
-
- 文章 · 前端 | 43分钟前 | 模块化 import ES6 export type="module"
- JavaScript模块导出与导入全面解析
- 425浏览 收藏
-
- 文章 · 前端 | 52分钟前 |
- JavaScriptRegExp使用技巧与实例详解
- 237浏览 收藏
-
- 文章 · 前端 | 1小时前 | 鼠标事件 跨浏览器兼容性 拖拽功能 transform属性 触摸设备支持
- JavaScript拖拽功能实现教程
- 449浏览 收藏
-
- 文章 · 前端 | 1小时前 | aria 事件委托 foreach openTab querySelectorAll
- JavaScript实现选项卡切换效果的技巧
- 487浏览 收藏
-
- 文章 · 前端 | 2小时前 | 密钥管理 数据加密 localStorage CryptoJS AES算法
- JavaScript本地存储加密技巧与实现
- 151浏览 收藏
-
- 文章 · 前端 | 2小时前 | 性能优化 requestAnimationFrame window.addEventListener debounce resize
- JavaScript监听浏览器窗口大小变化方法
- 269浏览 收藏
-
- 文章 · 前端 | 2小时前 |
- JavaScript严格模式详解与实用技巧
- 389浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 笔灵AI生成答辩PPT
- 探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
- 28次使用
-
- 知网AIGC检测服务系统
- 知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
- 42次使用
-
- AIGC检测-Aibiye
- AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
- 39次使用
-
- 易笔AI论文
- 易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
- 51次使用
-
- 笔启AI论文写作平台
- 笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
- 42次使用
-
- 优化用户界面体验的秘密武器:CSS开发项目经验大揭秘
- 2023-11-03 501浏览
-
- 使用微信小程序实现图片轮播特效
- 2023-11-21 501浏览
-
- 解析sessionStorage的存储能力与限制
- 2024-01-11 501浏览
-
- 探索冒泡活动对于团队合作的推动力
- 2024-01-13 501浏览
-
- UI设计中为何选择绝对定位的智慧之道
- 2024-02-03 501浏览