异步编程:从Promise到async/await的进化

前言

async/await 被引入之前,JavaScript 开发者主要依赖于回调函数和 Promise 来处理异步操作,如网络请求、文件读写等。尽管 Promise 相对于回调函数有了显著的改进,让异步代码更加结构化,但编写和阅读包含大量嵌套或链式 Promise 的代码仍然相对复杂,尤其是涉及错误处理时。async/await 便在 ES2017 中被引入,旨在进一步简化异步代码的编写,使其读起来更像是同步代码,从而提高代码的可读性和可维护性。

使用规范

当使用 async 关键字定义一个函数时,直接从async函数返回的非Promise值会被自动封装成 resolved 状态的 Promise,而直接返回的 Promise 本身的状态则取决于该 Promise 的处理逻辑。不了解Promise的同学可以先了阅读一下为什么我们需要Promise(一) - 掘金 (juejin.cn)学些Promise的作用和语法再阅读会更好理解。

自定义异步函数

javascript 复制代码
async function helloAsync() {
    return 'Hello, Async!';
}

helloAsync().then(message => console.log(message)); // 输出: Hello, Async!
javascript 复制代码
async function helloAsync() {
  return new Promise((reslove,reject)=>{
  })
}

helloAsync().then(message => console.log(message)); // 输出为空
javascript 复制代码
async function helloAsync() {
  return new Promise((reslove,reject)=>{
    reject('Error')
  })
}

helloAsync().then(()=>{},message => console.log(message)); // 输出:Error

异步等待

await 遇到一个 Promise(或 Promise-like 对象)时,它会使当前 async 函数的执行暂停,直到该 Promise 完成(解析或拒绝)。这意味着紧跟着在 await 后的代码不会立即执行,而是等待 Promise 的结果。

ini 复制代码
async function fetchData() {
    const response = await fetch('https://api.example.com/data');
    const data = await response.json();
    console.log(data);
}
fetchData();

并发执行

假设你需要同时从多个API获取数据,这些操作之间相互独立,不需要等待一个完成后再开始另一个。这时,使用 Promise.all() 可以并行执行这些请求,从而减少总等待时间。

javascript 复制代码
async function fetchMultipleResources() {
    const [usersResponse, postsResponse] = await Promise.all([
        fetch('https://api.example.com/users'),
        fetch('https://api.example.com/posts')
    ]);
}
fetchMultipleResources();

异常处理

由于async await没有像Promise一样链式调用catch的错误捕获不会机制,但是我们强调在 async 函数中使用 try...catch 来捕获 Promise 中的异常,提高错误处理的可读性和健壮性。

javascript 复制代码
async function fetchDataWithErrorHandling() {
    try {
        const response = await fetch('https://api.example.com/data');
        const data = await response.json();
        console.log(data);
    } catch (error) {
        console.error('An error occurred:', error);
    }
}

为什么需要 async/await

async/await 之前,开发者主要依靠 Promise 链式调用(.then().catch())来处理异步操作的结果。这种方式虽然解决了回调地狱的问题,但在处理多层嵌套的异步操作时,代码仍然可能变得难以理解和维护。例如,多个异步操作依次执行时,每个操作后都需要跟随着 .then(),这会导致代码层次很深,阅读和维护成本增加。

举个例子:我们需要从三个不同的 API 获取数据,并且每个请求必须等待前一个请求成功后才能开始。

使用传统的 Promise 链式调用:

javascript 复制代码
function fetchFirstData() {
  return new Promise((resolve, reject) => {
      setTimeout(() => resolve("First Data"), 1000); // 模拟异步操作
  });
}

function fetchSecondData() {
  return new Promise((resolve, reject) => {
      setTimeout(() => resolve("Second Data"), 1500); // 模拟异步操作
  });
}

function fetchThirdData() {
  return new Promise((resolve, reject) => {
      setTimeout(() => resolve("Third Data"), 2000); // 模拟异步操作
  });
}

fetchFirstData()
  .then(firstData => {
      console.log(firstData);
      return fetchSecondData();
  })
  .then(secondData => {
      console.log(secondData);
      return fetchThirdData();
  })
  .then(thirdData => {
      console.log(thirdData);
  })
  .catch(error => {
      console.error("An error occurred:", error);
  });

使用 async/await 的例子:

javascript 复制代码
async function fetchAllData() {
  try {
      const firstData = await fetchFirstData();
      console.log(firstData);
      const secondData = await fetchSecondData();
      console.log(secondData);
      const thirdData = await fetchThirdData();
      console.log(thirdData);
  } catch (error) {
      console.error("An error occurred:", error);
  }
}

// 调用这个异步函数
fetchAllData();

总结

上述例子可以明显的看到在 fetchAllData 函数中,通过 await 关键字,我们可以按顺序等待每个异步操作完成,而无需嵌套 .then(),这让代码看起来就像是同步执行一样,极大地提高了可读性。错误处理也变得更加集中,使用一个 try...catch 块即可捕获整个过程中的任何错误。只有等前面的异步操作完成后,后面的异步操作才能继续,这样的代码可读性和可维护性也是最高的

相关推荐
小鑫同学9 分钟前
Alias Assistant:新一代 macOS Shell 别名管理解决方案
前端·前端工程化
꒰ঌ小武໒꒱9 分钟前
RuoYi-Vue 前端环境搭建与部署完整教程
前端·javascript·vue.js·nginx
名字越长技术越强27 分钟前
前端之相对路径
前端
望道同学1 小时前
PMP/信息系统项目管理师 9 张 思维导图【考试必备】
前端·后端·程序员
局i2 小时前
Vue 中 v-text 与 v-html 的区别:文本渲染与 HTML 解析的抉择
前端·javascript·vue.js
fruge2 小时前
接口 Mock 工具对比:Mock.js、Easy Mock、Apifox 的使用场景与配置
开发语言·javascript·ecmascript
菜鸟冲锋号2 小时前
问题:增量关联(实时同步新数据) 这个场景中,如果hudi_pay 变更了一条数据,hudi_order_pay_join 结果的数据会跟着变化吗
服务器·前端·数据库
贩卖黄昏的熊2 小时前
typescript 快速入门
开发语言·前端·javascript·typescript·ecmascript·es6
拾柒SHY3 小时前
XSS-Labs靶场通关
前端·web安全·xss
前端婴幼儿3 小时前
前端主题切换效果
前端