热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

深入理解async/await:优雅的异步编程

async/await是现代JavaScript中非常强大的异步编程工具,可以极大地简化异步代码的编写。本文将详细介绍async和await的用法及其背后的原理。

### 引言

async/await 是 Javascript 中非常强大的异步编程工具,可以极大地简化异步代码的编写。通过 async 和 await 关键字,我们可以写出更加直观和易读的异步代码。本文将详细介绍 async 和 await 的用法及其背后的原理。

异步编程示例

### async 的作用

async 关键字用于声明一个函数是异步的,该函数会返回一个 Promise 对象。可以在 async 函数内部使用 then 方法添加回调函数。async 函数内部的 return 语句返回的值会成为 then 方法回调函数的参数。

async function test() { return 'test'; }
console.log(test); // [AsyncFunction: test] async 函数是 [AsyncFunction] 构造函数的实例
console.log(test()); // Promise { 'test' } // async 返回的是一个 Promise 对象
test().then(res => { console.log(res); // test });
// 如果 async 函数没有返回值,async 函数返回一个 undefined 的 Promise 对象
async function fn() { console.log('没有返回'); }
console.log(fn()); // Promise { undefined }
// 可以看到 async 函数返回值和 Promise.resolve() 一样,将返回值包装成 Promise 对象,如果没有返回值就返回 undefined 的 Promise 对象

### await 的用法

await 操作符只能在异步函数 async function 内部使用。如果一个 Promise 被传递给一个 await 操作符,await 将等待 Promise 正常处理完成并返回其处理结果,也就是说它会阻塞后面的代码,等待 Promise 对象结果。如果等待的不是 Promise 对象,则返回该值本身。

async function test() { return new Promise((resolve) => { setTimeout(() => { resolve('test 1000'); }, 1000); }); }
function fn() { return 'fn'; }
async function next() { let res0 = await fn(), res1 = await test(), res2 = await fn(); console.log(res0); console.log(res1); console.log(res2); }
next(); // 1s 后才打印出结果,因为 res1 在等待 Promise 的结果,阻塞了后面代码。

### 错误处理

如果 await 后面的异步操作出错,那么等同于 async 函数返回的 Promise 对象被 reject。

async function test() { await Promise.reject('错误了'); }
test().then(res => { console.log('success', res); }, err => { console.log('err ', err); }) // err 错误了
// 防止出错的方法是将其放在 try...catch 代码块之中
async function test() { try { await new Promise(function (resolve, reject) { throw new Error('错误了'); }); } catch (e) { console.log('err', e); } return await('成功了'); }

### 并行执行多个 await

多个 await 命令后面的异步操作,如果不存在继发关系(即互不依赖),最好让它们同时触发。

let foo = await getFoo();
let bar = await getBar();
// 上面这样写法 getFoo 完成以后,才会执行 getBar
// 同时触发写法 ↓
// 写法一
let [foo, bar] = await Promise.all([getFoo(), getBar()]);
// 写法二
let fooPromise = getFoo();
let barPromise = getBar();
let foo = await fooPromise;
let bar = await barPromise;

### async/await 的优点

async/await 的优势在于处理由多个 Promise 组成的 then 链。在之前的 Promise 文章中提到,用 then 处理回调地狱的问题,async/await 相当于对 Promise 的进一步优化。假设一个业务,分多个步骤,且每个步骤都是异步的,而且依赖上个步骤的执行结果。

// 假设表单提交前要通过两个校验接口
async function check(ms) { // 模仿异步 return new Promise((resolve) => { setTimeout(() => { resolve(`check ${ms}`); }, ms); }); }
function check1() { console.log('check1'); return check(1000); }
function check2() { console.log('check2'); return check(2000); }
// -------------promise------------
function submit() { console.log('submit'); // 经过两个校验,多级关联,promise 传值嵌套较深 check1().then(res1 => { check2(res1).then(res2 => { /* 提交请求 */ }) }) }
submit();
// -------------async/await-----------
async function asyncAwaitSubmit() { let res1 = await check1(), res2 = await check2(res1); console.log(res1, res2); /* 提交请求 */ }

### 原理

async 函数的实现原理,就是将 Generator 函数和自动执行器,包装在一个函数里。

async function fn(args) { // ... }
// 等同于
function fn(args) { return spawn(function* () { // ... }); }

### Generator 函数的用法

Generator 函数就是一个封装的异步任务,或者说是异步任务的容器。异步操作需要暂停的地方,都用 yield 语句注明。调用 Generator 函数,返回的是指针对象(这是它和普通函数的不同之处)。调用指针对象的 next 方法,会移动内部指针。next 方法的作用是分阶段执行 Generator 函数。每次调用 next 方法,会返回一个对象,表示当前阶段的信息(value 属性和 done 属性)。value 属性是 yield 语句后面表达式的值,表示当前阶段的值;done 属性是一个布尔值,表示 Generator 函数是否执行完毕,即是否还有下一个阶段。

function* Generator() { yield '1'; yield Promise.resolve(2); return 'ending'; }
var gen = Generator(); // 返回指针对象 Object [Generator] {}
let res1 = gen.next();
console.log(res1); // 返回当前阶段的值 { value: '1', done: false }
let res2 = gen.next();
console.log(res2); // 返回当前阶段的值 { value: Promise { 2 }, done: false }
res2.value.then(res => { console.log(res); // 2 });
let res3 = gen.next();
console.log(res3); // { value: 'ending', done: true }
let res4 = gen.next();
console.log(res4); // { value: undefined, done: true }

### 使用 Generator 实现 async 函数

// 接受一个 Generator 函数作为参数
function spawn(genF) { // 返回一个函数 return function() { // 生成指针对象 const gen = genF.apply(this, arguments); // 返回一个 Promise return new Promise((resolve, reject) => { // key 有 next 和 throw 两种取值,分别对应了 gen 的 next 和 throw 方法 // arg 参数则是用来把 Promise resolve 出来的值交给下一个 yield function step(key, arg) { let result; // 监控到错误 就把 Promise 给 reject 掉 外部通过 .catch 可以获取到错误 try { result = gen[key](arg); } catch (error) { return reject(error); } // gen.next() 返回 { value, done } 的结构 const { value, done } = result; if (done) { // 如果已经完成了 就直接 resolve 这个 Promise return resolve(value); } else { // 除了最后结束的时候外,每次调用 gen.next() return Promise.resolve( // 这个 value 对应的是 yield 后面的 Promise value ).then((val) => step('next', val), (err) => step('throw', err)); } } step('next'); }); }}

### 测试

function fn(nums) { return new Promise(resolve => { setTimeout(() => { resolve(nums); }, 1000); }); }
// async 函数
async function testAsync() { let res1 = await fn(1); console.log(res1); // 1 let res2 = await fn(2); console.log(res2); // 2 return res2; }
let _res = testAsync();
console.log('testAsync-res', _res); // Promise
_res.then(v => console.log('testAsync-res', v)); // 2
// Generator 函数
function* gen() { let res1 = yield fn(3); console.log(res1); // 3 let res2 = yield fn(4); console.log(res2); // 4 // let res3 = yield Promise.reject(5); // console.log(res3); return res2; }
let _res2 = spawn(gen)();
console.log('gen-res', _res2); // Promise
_res2.then(v => console.log('gen-res', v)) // 4
.catch(err => { console.log(err); }); // res3 执行会抛出异常

### 结语

如果你觉得此文对你有一丁点帮助,点个赞。或者可以加入我的开发交流群:1025263163,相互学习,我们会有专业的技术答疑解惑。

如果你觉得这篇文章对你有点用的话,麻烦请给我们的开源项目点点 star: https://gitee.com/ZhongBangKeJi/CRMEB 不胜感激!


推荐阅读
  • C++ 异步编程中获取线程执行结果的方法与技巧及其在前端开发中的应用探讨
    本文探讨了C++异步编程中获取线程执行结果的方法与技巧,并深入分析了这些技术在前端开发中的应用。通过对比不同的异步编程模型,本文详细介绍了如何高效地处理多线程任务,确保程序的稳定性和性能。同时,文章还结合实际案例,展示了这些方法在前端异步编程中的具体实现和优化策略。 ... [详细]
  • 重要知识点有:函数参数默许值、盈余参数、扩大运算符、new.target属性、块级函数、箭头函数以及尾挪用优化《深切明白ES6》笔记目次函数的默许参数在ES5中,我们给函数传参数, ... [详细]
  • 开发日志:高效图片压缩与上传技术解析 ... [详细]
  • 在尝试对 QQmlPropertyMap 类进行测试驱动开发时,发现其派生类中无法正常调用槽函数或 Q_INVOKABLE 方法。这可能是由于 QQmlPropertyMap 的内部实现机制导致的,需要进一步研究以找到解决方案。 ... [详细]
  • 本指南介绍了如何在ASP.NET Web应用程序中利用C#和JavaScript实现基于指纹识别的登录系统。通过集成指纹识别技术,用户无需输入传统的登录ID即可完成身份验证,从而提升用户体验和安全性。我们将详细探讨如何配置和部署这一功能,确保系统的稳定性和可靠性。 ... [详细]
  • PHP预处理常量详解:如何定义与使用常量 ... [详细]
  • 在深入研究 UniApp 封装请求时,发现其请求 API 方法中使用了 `then` 和 `catch` 函数。通过详细分析,了解到这些函数是 Promise 对象的核心组成部分。Promise 是一种用于处理异步操作的结果的标准化方式,它提供了一种更清晰、更可控的方法来管理复杂的异步流程。本文将详细介绍 Promise 的基本概念、结构和常见应用场景,帮助开发者更好地理解和使用这一强大的工具。 ... [详细]
  • 在Delphi7下要制作系统托盘,只能制作一个比较简单的系统托盘,因为ShellAPI文件定义的TNotifyIconData结构体是比较早的版本。定义如下:1234 ... [详细]
  • 单元测试:使用mocha和should.js搭建nodejs的单元测试
    2019独角兽企业重金招聘Python工程师标准BDD测试利器:mochashould.js众所周知对于任何一个项目来说,做好单元测试都是必不可少 ... [详细]
  • 本地存储组件实现对IE低版本浏览器的兼容性支持 ... [详细]
  • 本文介绍了如何使用 Node.js 和 Express(4.x 及以上版本)构建高效的文件上传功能。通过引入 `multer` 中间件,可以轻松实现文件上传。首先,需要通过 `npm install multer` 安装该中间件。接着,在 Express 应用中配置 `multer`,以处理多部分表单数据。本文详细讲解了 `multer` 的基本用法和高级配置,帮助开发者快速搭建稳定可靠的文件上传服务。 ... [详细]
  • 技术分享:使用 Flask、AngularJS 和 Jinja2 构建高效前后端交互系统
    技术分享:使用 Flask、AngularJS 和 Jinja2 构建高效前后端交互系统 ... [详细]
  • 本文将继续探讨 JavaScript 函数式编程的高级技巧及其实际应用。通过一个具体的寻路算法示例,我们将深入分析如何利用函数式编程的思想解决复杂问题。示例中,节点之间的连线代表路径,连线上的数字表示两点间的距离。我们将详细讲解如何通过递归和高阶函数等技术实现高效的寻路算法。 ... [详细]
  • 本文介绍了如何利用ObjectMapper实现JSON与JavaBean之间的高效转换。ObjectMapper是Jackson库的核心组件,能够便捷地将Java对象序列化为JSON格式,并支持从JSON、XML以及文件等多种数据源反序列化为Java对象。此外,还探讨了在实际应用中如何优化转换性能,以提升系统整体效率。 ... [详细]
  • 在 Linux 环境下,多线程编程是实现高效并发处理的重要技术。本文通过具体的实战案例,详细分析了多线程编程的关键技术和常见问题。文章首先介绍了多线程的基本概念和创建方法,然后通过实例代码展示了如何使用 pthreads 库进行线程同步和通信。此外,还探讨了多线程程序中的性能优化技巧和调试方法,为开发者提供了宝贵的实践经验。 ... [详细]
author-avatar
mobiledu2402851323
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有