消灭异步回调,还得是 async-await

关于异步处理问题,ES5的回调让我们陷入回调地狱轮回,后来ES6的Promise(Promise不了解?点这了解[1])让我们脱离轮回,终于,ES7的async-await带我们走向光明。今天我们就来学习一夏 async-await,看看与Promise有何联系和区别。

一、走进Async-await原理
1、原理1

async函数返回一个 Promise 对象,可以使用then方法添加回调函数。举例说明:

// async返回的是Promise对象?
async function testAsync() {return 'hello';//上篇文章Promise对象的返回值如果不是Promise,会通过Promise.resolve()转化为Promise,再进行处理
}
const result = testAsync()
console.log(result);//Promise { 'hello' }  说明async返回的是Promise对象
复制代码

那既然async返回的是Promise对象,那么async后面的函数可以接.then()或者.catch()...嘛?我们试一试就知道了。

// async返回的是Promise对象,并且可以接Promise的方法?
async function testAsync() {// await await等待还是promise对象return 'hello'
}
testAsync().then((result)=>{console.log(result);}).catch((error)=>{console.log(error);})
//hello  妈耶!打印了!说明async返回的是Promise对象,并且可以接Promise的方法,并且!!!默认状态是resolved的
复制代码

上面代码说明,async函数内部return语句返回的值,会成为then方法回调函数的参数

2、原理2

当async函数内部抛出错误的时候,会导致返回的 Promise 对象变为reject状态。抛出的错误对象会被.then()方法的第二个回调函数接收或者.catch()方法回调函数接收到。

// async函数内部抛出错误或者Promise状态为reject
async function testError(){//throw new Error('出错啦~~');await Promise.reject('出错了');//await前面有return和没有return效果一样
} 
testError()// .then(()=>{},(error)=>{console.log(error);}).catch(error=>{console.log(error);})
//Error: 出错啦~~
复制代码
3、原理3

await命令后面是一个 Promise 对象,返回该对象的结果。如果不是 Promise 对象,就直接返回对应的值。代码说明:

// await
async function getName(){// return '来自星星的你';return await '来自星星的你';//上面直接return等价于这个return
}
getName().then(result=>{console.log(result);})
//来自星星的你
复制代码
4、原理4

await的使用,必须要有async。这便是async-await的浪漫所在了:async返回的是一个Promise对象,await等待的就是这个Promise对象,所以await不能没有async(但是async可以没有await)。有没有被浪漫到?反正我是醉了。如果await没有async会怎么样?报错:

// await没有async会报错
function testAwait(){return await '西红柿炒辣椒'
}
testAwait().catch(error=>{console.log(error);})
//SyntaxError: await is only valid in async function
复制代码
二、深入Async-await规则
1、async封装Promise

 

// async封装Promise
async function fn1() {return '喜羊羊与灰太狼';// //相当于return Promise.resolve('喜羊羊与灰太狼')const data = await fn1();//接收data值
}
fn1()//执行async函数,返回的是一个Promise对象.then(data => {console.log('content =', data)})//content = 喜羊羊与灰太狼
复制代码
2、await相当于then
// await---.then()
async function getName(){const operate=Promise.resolve('白雪公主')//执行函数const name= await operate //await相当于Promise的then  operate.then(name=>{})console.log('name:',name)
}
getName();
( async function(){const person=await '七个小矮人' //await Promise.resolve('七个小矮人') await后面不跟Promise,也会被封装成Promiseconsole.log('person:',person)//400
})();//自执行函数//name: 白雪公主
//person: 七个小矮人
复制代码
3、多个await时,按时序执行

当函数执行的时候,一旦遇到await就会先返回,等到异步操作完成,再接着执行函数体内后面的语句。任何一个await语句后面的 Promise 对象变为reject状态,那么整个async函数都会中断执行

async function testOrder() {await Promise.reject('出错了')//UnhandledPromiseRejectionWarning: 出错了await Promise.resolve('hello world'); // 不会执行
}
testOrder();
复制代码
4、try…catch相当于catch

如果希望即使前一个异步操作失败,也不要中断后面的异步操作。可将第一个await放在try...catch结构里面,这样不管这个异步操作是否成功,第二个await都会执行。

// try...catch
!(async function () {const testError = Promise.reject('出错啦~~~')//rejected状态// const testError=throw new Error('出错啦~~~');try {const result = await testError; //await相当于then,但是reject不会触发thenconsole.log('success:'+result) //不会输出,因为const result = await testError被报错,被catch捕获} catch (error) {console.error('error:'+error)//try...catch 相当于Promise的catch}})()
//error:出错啦~~~
复制代码

当await后面是Promise对象的时候,我们也可直接在await后面直接.catch捕获错误:

 

async function testError() {await Promise.reject('出错了').catch(error => console.log(error));//这里捕获错误,不会影响下一个await执行return await Promise.resolve('hello world');
}testError().then(result => console.log(result))
复制代码
三、解析Async-await语法

我们浅浅看一个面试题:

// 面试题
function getJSON() {return new Promise((resolve, reject) => {setTimeout(() => {console.log(2);resolve(2)}, 2000)})
}
async function testAsync() {await getJSON()console.log(3);
}
testAsync()
//2
//3
复制代码

问题当然不会问打印顺序啦,问题是将async await语句解析翻译为Promise?

根据现在的知识面,我们必须知道:

(1)await不能单独出现,其函数前面一定要有async。

(2)await会干两件事:

第一,将写在await后面的代码放到async创建的那个Promise里面执行。

第二、将写在await下面的代码放到前一个创建的那个Promise对象的.then里面执行。

(3)await返回的也是Promise对象,他只是把await下面的代码放到了await返回的promise的.then里面执行。

这样的话,是不是如鱼得水了。翻译如下:

function getJSON() {return new Promise((resolve, reject) => {setTimeout(() => {console.log(2);resolve(2)}, 2000)})
}
// 编译成Promise原理
function testAsync() {return Promise.resolve().then(() => {return getJSON();}).then(() => {console.log(3);})
}
testAsync()
复制代码
四、拓展Async-await应用
1、场景1

你学废async-await了嘛?还记得上一篇开篇的回调地狱嘛?我们通过Promise解决回调是这样的:

// Promise解决方式
function doCallback(n) {var myPromise = new Promise(function (resolve, reject) {   //处理异步任务var flag = true;setTimeout(function () {if (flag) {resolve(n)}else {reject('失败')}},0)})return myPromise;
}doCallback(1).then((result) => { //then是成功执行的方法 返回的还是一个Promise对象console.log(result);//打印张三  res是执行return fn(2);}).then((result) => {console.log(result);return fn(3)}).then((result) => {console.log(result);return fn(4)}).then((result) => {console.log(result);}).catch((result) => { //catch是失败执行的方法console.log(result);})//好多.then,形成.then链啦
//1
//2
//3
//4
复制代码

通过以上Promise方法,可以明显解决回调地狱“向右移”的浮夸表现,但是,Promise是基于 then, catch 的链式调用,但也是基于回调函数。.then链多多少少还是违背原生代码,显得也不是很优雅。作为回调终极武器,async-await更加贴近于原生代码,我们看一下吧:

//封装一个返回promise的异步任务
function doCallback(str) {var myPromise = new Promise(function (resolve, reject) {var flag = true;setTimeout(function () {if (flag) {resolve(str)} else {reject('处理失败')}})})return myPromise;
}//封装一个执行上述异步任务的async函数
async function testAsync() {var result1 = await doCallback(1);  //await直接拿到fn()返回的promise的数据,并且赋值给resultvar result2 = await doCallback(2);  //await 后面的代码,都可以看做是异步回调 callback 里的内容,都是异步的var result3 = await doCallback(3);var result4 = await doCallback(4);console.log(result1);console.log(result2);console.log(result3);console.log(result4);
}//这样是不是简洁优雅多了呢?
//执行函数
testAsync();
//1
//2
//3
//4
复制代码

有了 async-await、promise 还有必要学习吗?通过上面async-await的解决方案可以瞧见,async / await 和 Promise 并不互斥,二者相辅相成。同时async / await 并不能改变异步的本质( js是单线程的,异步需要回调,都是要基于 event loop 来实现(什么是event loop?关注我,等我文章~))。

总结

现在知道了,async-await是promise的语法糖了吧,不仅让我们书写代码时更加流畅,而且增强了代码的可读性。特别注意的是:虽然async-await 是建立在 Promise机制之上的,但是并不能取代其地位,他们两者相辅相成,息息相关。其实async-await不止是Promise的语法糖,还是Generator的语法糖

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/61084.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

E9—TEMAC IP实现千兆网口UDP传输2023-08-28

1.关于IP收费的问题 Tri Mode Ethernet MAC是收费IP,打开IP后,当左下角显示Bought IP license available则IP可用。 2.功能说明 应用搭建的场景是,上位机发送数据,首先发起arp请求,随后下位机给出arp应答响应&#…

说说我最近筛简历和面试的感受。。

大家好,我是鱼皮。 都说现在行情不好、找工作难,但招人又谈何容易?! 最近我们公司在招开发,实习社招都有。我收到的简历很多,但认真投递的、符合要求的却寥寥无几,而且都是我自己看简历、选人…

【Unity-Cinemachine相机】虚拟相机(Virtual Camera)的本质与基本属性

我们可以在游戏进行时修改各个属性,但在概念上,最好将Virtual Camera 当作一种相机行为的“配置文件”,而不是一个组件。 我们的相机有几种行为就为它准备几种虚拟相机,比如角色移动就为它第三人称相机,瞄准就准备一个…

驾驶员监控系统DMS系统功能规范

概述 文档范围 该文档阐述了DMS系统的功能场景、系统组成、接口需求等。 目的 该功能规范为DMS系统及周边件的开发提供参考。 缩写与定义 英文缩写 英文全称 中文描述 DMS: Driver Monitoring System 驾驶员监控系统 MPU: Micro Processor Unit 微处理器单元 IECU: …

软技能的重要性:在面试中展示团队合作与沟通能力

🌷🍁 博主猫头虎 带您 Go to New World.✨🍁 🦄 博客首页——猫头虎的博客🎐 🐳《面试题大全专栏》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺 &a…

【ctf题目系列】ctfwiki pwn类型

categories: ctf ret2shellcode [rootningan ret2shellcode]# ./ret2shellcode No system for you this time !!! 123 bye bye ~[rootningan ret2shellcode]#checksec检查 [rootningan ret2shellcode]# checksec ret2shellcode [!] Could not populate PLT: future feature …

MySQL基本语法

基础语法 别名 -- 别名 as 可省略,但中间需要空格 select name as 员工姓名, position as 职位名称 from employees;常量和运算 select 200, 篮球 as hobby; select order_id, unit_price, quantity, unit_price * quantity as total_amount from orders;运算符 …

CNN 02(CNN原理)

一、卷积神经网络(CNN)原理 1.1 卷积神经网络的组成 定义 卷积神经网络由一个或多个卷积层、池化层以及全连接层等组成。与其他深度学习结构相比,卷积神经网络在图像等方面能够给出更好的结果。这一模型也可以使用反向传播算法进行训练。相比较其他浅层或深度神经…

Mysql-InnoDB记录结构

一、InnoDB简介 InnoDB 采取的方式是:将数据划分为若干个页,以页作为磁盘和内存之间交互的基本单位,InnoDB中页的大小一般为 16 KB。也就是在一般情况下,一次最少从磁盘中读取16KB的内容到内存中,一次最少把内存中的1…

快速学会git版本管理——基础命令

1、初始化仓库: git init 2.克隆一个仓库: git clone 我的仓库 3.添加文件到暂存区: git add 文件名 如果要上传所有文件 可以用:add . 中间要有空格隔开 4.提交更改: git commit -m "提交" 如果要添加对跟踪文件所做的所…

编程题四大算法思想(二)——回溯法:N皇后问题、子集和问题、地图填色问题、迷宫问题

文章目录 回溯法迷宫游戏 N皇后问题基本概念解空间4后问题的解空间 可行解和最优解回溯法回溯法术语回溯法的关键问题回溯法的基本思想4后问题的约束条件n后问题生成问题状态的基本方法 子集和问题一个朴素的求解方法回溯回溯法的剪枝技术 地图填色问题 回溯法 迷宫游戏 深度优…

Virtual

虚拟接口可以用作编写操作系统和驱动程序独立测试的一种方式。任何连接到同一通道(来自同一Python进程)的VirtualBus实例都将相互接收消息。 如果消息应跨进程或主机边界发送,请考虑使用多播IP接口,并参考虚拟接口对不同虚拟接口进行比较和一般性讨论。 Example import …

使用 Python编程: 下载 YouTube 音频的桌面应用程序

最近我开发了一个使用 Python 编写的桌面应用程序,可以方便地下载 YouTube 音频。该应用程序使用了 wxPython、yt_dlp 和 tqdm 库,提供了一个简单直观的用户界面,并具备高效的下载功能。 C:\pythoncode\new\youtube-dl-audio.py 程序介绍 …

为了他的鸟,做件很叛逆很酷的事儿

有种鸟儿,叫隐鹮(Geronticus eremita),大小如鹅,头部光秃,嘴巴巨大,一个字,丑。可是,它还有一个特点,面临濒危。 为了能在欧洲冬季存活,这种鸟儿需…

webassembly003 ggml GGML Tensor Library part-2 官方使用说明

https://github.com/ggerganov/whisper.cpp/tree/1.0.3 GGML Tensor Library 官方有一个函数使用说明,但是从初始版本就没修改过 : https://github1s.com/ggerganov/ggml/blob/master/include/ggml/ggml.h#L3-L173 This documentation is still a work in progres…

安达发|富士康科技集团利用自动排程APS软件打造智慧工厂

富士康科技集团作为全球领先的3C产品研发制造企业,近年来积极布局智能制造领域,通过引入先进的自动化排程系统(APS),成功打造了智慧工厂,提高了生产质量与效率,降低了生产成本。 富士康集团自2019年下半年提出在观澜厂区建立数字可…

DRM全解析 —— CREATE_DUMB(2)

接前一篇文章:DRM全解析 —— CREATE_DUMB(1) 本文参考以下博文: DRM驱动(三)之CREATE_DUMB 特此致谢! 上一回围绕libdrm与DRM在Linux内核中的接口: DRM_IOCTL_DEF(DRM_IOCTL_MOD…

C++ list模拟实现

list模拟实现代码&#xff1a; namespace djx {template<class T>struct list_node{T _data;list_node<T>* _prev;list_node<T>* _next;list_node(const T& x T()):_data(x),_prev(nullptr),_next(nullptr){}};template<class T,class Ref,class Pt…

Linux 云服务器挂载数据盘

1、检查linux服务器磁盘情况 df -h 可以看到无磁盘挂载信息。 2、查看待挂载磁盘信息 fdisk -l 可以看到40G系统盘、50G数据盘&#xff08;盘符&#xff1a;/dev/vdb&#xff09; 3、对数据盘分区 fdisk /dev/vdb 根据提示&#xff0c;依次输入“n”&#xff0c;“p”“1…

今日举办亚马逊云科技re:Inforce大会:结合生成式AI安全最新趋势,解决企业新挑战

2023年&#xff0c;生成式AI带来了无数的创新&#xff0c;并将会在行业应用中产生更多的新能力、新场景。与此同时&#xff0c;关于生成式AI的风险管控成为各方关注焦点&#xff0c;数据隐私、合规保护、防欺诈等&#xff0c;已成为生成式AI时代的安全合规的新话题。 随着云上业…