掌握 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 中如何调试匿名函数?
- 下一篇
- Golang 函数如何影响程序性能?
-
- 文章 · 前端 | 5分钟前 | CSS 隐藏 :empty 空元素 :only-child
- CSS空元素隐藏技巧:empty与only-child组合应用
- 176浏览 收藏
-
- 文章 · 前端 | 8分钟前 |
- CSS文件过多怎么优化?合并策略详解
- 349浏览 收藏
-
- 文章 · 前端 | 17分钟前 |
- 纯HTML代码怎么运行【教程】
- 230浏览 收藏
-
- 文章 · 前端 | 22分钟前 |
- HTML代码部署步骤详解
- 193浏览 收藏
-
- 文章 · 前端 | 29分钟前 | html JavaScript 浏览器 页面渲染 DOM树
- HTML运行原理揭秘|网页加载全过程解析
- 232浏览 收藏
-
- 文章 · 前端 | 34分钟前 |
- 前端组件文档搭建指南
- 415浏览 收藏
-
- 文章 · 前端 | 35分钟前 |
- JavaScriptTemporalAPI详解与使用教程
- 282浏览 收藏
-
- 文章 · 前端 | 37分钟前 |
- 前端自动化部署流程全解析
- 208浏览 收藏
-
- 文章 · 前端 | 38分钟前 | jwt OAuth XSS攻击 HttpOnlyCookie 安全存储
- JWT与OAuth安全存储方法解析
- 188浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3179次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3390次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3418次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4525次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3798次使用
-
- JavaScript函数定义及示例详解
- 2025-05-11 502浏览
-
- 优化用户界面体验的秘密武器:CSS开发项目经验大揭秘
- 2023-11-03 501浏览
-
- 使用微信小程序实现图片轮播特效
- 2023-11-21 501浏览
-
- 解析sessionStorage的存储能力与限制
- 2024-01-11 501浏览
-
- 探索冒泡活动对于团队合作的推动力
- 2024-01-13 501浏览

