理解ES7中的async/await
理解ES7中的async/await
优势是:就是解决多层异步回调的嵌套
从字面上理解 async/await, async是 "异步"的含义,await可以认为是 async wait的简写,因此可以理解 async 用于声明一个function是异步的,而await用于等待一个异步方法执行完成返回的值(返回值可以是一个Promise对象或普通返回的值)。
注意:await 只能出现在 async函数中。
1-1 async的作用?
首先来理解async函数是怎么处理返回值的,我们以前写代码都是通过return语句返回我们想要的值,但是async与return有什么区别呢? 下面我们看如下代码:
async function testAsync() {
return 'hello world';
}
const result = testAsync();
console.log(result); // 打印出来看到 返回一个promise对象
如上代码,我们输出的是一个Promise对象。因此 async返回的是一个Promise对象,因此我们可以使用 then()链来处理这个Promise对象。像如下代码:
async function testAsync() {
return 'hello world';
}
const result = testAsync();
result.then(v => {
console.log(v); // 输出 hello world
});
但是如果 async函数没有返回值的话,那么就会返回 undefined; 如下代码:
async function testAsync() {
return;
}
const result = testAsync();
result.then(v => {
console.log(v); // 输出 undefined
});
1-2 await的作用?
从语义上说,await是在等待一个async函数完成的,async函数返回的是一个Promise对象,await等待的是一个表达式,这个表达式的计算结果是Promise对象或其他值。
async函数返回一个Promise对象,await作用是用于等待一个async函数的返回值。
await函数后面可以接普通函数调用或直接量,请看如下代码:
function getSomething() {
return 'something';
}
async function testAsync() {
return Promise.resolve('hello');
}
async function test () {
const v1 = await getSomething();
const v2 = await testAsync();
console.log(v1); // something
console.log(v2); // hello
}
test();
注意:await 是等待async函数返回的Promise对象或其他值,await是一个运算符,用于组成表达式, 如果等待的是一个Promise对象,await会阻塞后面的代码(async调用不会造成堵塞,它内部所有的堵塞
都被封装在一个Promise对象中异步执行),等待Promise对象的resolve,然后得到resolve的值,作为await表达式的运输结果。
1-3 async/await 的一起使用的作用及优势在哪?
我们先不用 async/await, 来看看使用setTimeout模拟异步操作如下代码:
var test = function(time) {
return new Promise((resolve, reject) => {
setTimeout(function(){
resolve('hello world');
}, time)
});
};
test(1000).then((v) => {
console.log(v);
});
如果改用 async/await 代码如下:
var testAsync = function(time) {
return new Promise((resolve, reject) => {
setTimeout(function(){
resolve('hello world');
}, time)
});
};
async function test(time) {
const v = await testAsync(time);
console.log(v); // hello world
}
test(1000);
看上面的代码,反而会觉得 使用 async/await 代码变得多一点,复杂一点,但是结果貌似都一样,那么使用 async/await的优势在哪?
1-4 async/await的优势在于处理then链
单一的Promise链貌似不能发现 async/await的优势,但是如果需要处理多个Promise组成的then链的时候,优势可以看出来,因为Promise是通过then链来解决多层回调的问题,现在我们又可以使用
async/await来进一步优化,他们的优点就是解决多层异步回调的嵌套。
假设我们现在有一个需求是,分多个步骤完成,每个步骤都是异步的,并且后面的异步都需要依赖于上一个异步回调返回的数据,进行往下传递。我们先用 setTimeout来模拟异步操作。
function useTimeout (t) {
return new Promise((resolve, reject) => {
setTimeout(()=> {
resolve(t+100)
}, t);
});
}
function step1(t) {
console.log(`step1 with ${t}`); // step1 with 300
return useTimeout(t);
}
function step2(t) {
console.log(`step2 with ${t}`); // step2 with 400
return useTimeout(t);
}
function step3(t) {
console.log(`step3 with ${t}`); // step3 with 500
return useTimeout(t);
}
function test() {
const time1 = 300;
step1(time1)
.then(time2 => step2(time2))
.then(time3 => step3(time3))
.then(res => {
console.log(`result is ${res}`); // result is 600
})
}
test();
如果我们使用 async/await来实现,代码变为如下:
function useTimeout (t) {
return new Promise((resolve, reject) => {
setTimeout(()=> {
resolve(t+100)
}, t);
});
}
function step1(t) {
console.log(`step1 with ${t}`); // step1 with 300
return useTimeout(t);
}
function step2(t) {
console.log(`step2 with ${t}`); // step2 with 400
return useTimeout(t);
}
function step3(t) {
console.log(`step3 with ${t}`); // step3 with 500
return useTimeout(t);
}
async function test() {
const time1 = 300;
const time2 = await step1(time1);
const time3 = await step2(time2);
const result = await step3(time3);
console.log(`result is ${result}`);
}
test();
上面我们可以看到 使用async/await 代码看起来都是同步的,等第一步完成后,再执行第二步,依次类推..... 并且不需要更多的回调函数嵌套。
下面我们再来看下我们之前讲的,后面的步骤需要上一个步骤的结果传递进去,使用async/await的优势可能更明显。
如下代码:
function useTimeout (t) {
return new Promise((resolve, reject) => {
setTimeout(()=> {
resolve(t+100)
}, t);
});
}
function step1(t1) {
console.log(`step1 with ${t1}`); // step1 with 300
return useTimeout(t1);
}
function step2(t1, t2) {
console.log(`step2 with ${t1} and ${t2}`); // step2 with 300 and 400
return useTimeout(t1+t2);
}
function step3(t1, t2, t3) {
console.log(`step3 with ${t1} and ${t2} and ${t3}`); // step3 with 300 and 400 and 800
return useTimeout(t1+t2+t3);
}
async function test() {
const time1 = 300;
const time2 = await step1(time1);
const time3 = await step2(time1, time2);
const result = await step3(time1, time2, time3);
console.log(`result is ${result}`); // result is 1600
}
test();
1-5 捕捉错误
我们可以使用 try catch 来捕获错误,如下代码:
var sleep = function (time) {
return new Promise((resolve, reject) => {
setTimeout(function() {
// 出错了, 返回 error
reject('error');
}, time);
})
};
var start = async function() {
try {
console.log('start');
await sleep(3000); // 返回一个错误
// 下面代码不会被执行
console.log('end');
} catch (err) {
console.log(err); // 捕捉错误 error
}
};
start();
理解ES7中的async/await的更多相关文章
- 理解C#中的 async await
前言 一个老掉牙的话题,园子里的相关优秀文章已经有很多了,我写这篇文章完全是想以自己的思维方式来谈一谈自己的理解.(PS:文中涉及到了大量反编译源码,需要静下心来细细品味) 从简单开始 为了更容易理解 ...
- JavaScript ES7 中使用 async/await 解决回调函数嵌套问题
原文链接:http://aisk.me/using-async-await-to-avoid-callback-hell/ JavaScript 中最蛋疼的事情莫过于回调函数嵌套问题.以往在浏览器中, ...
- 关于ES7中的async/await在客户端和服务端上的实践
一.前言 在项目中经常遇到处理异步请求的情况,面对层层的嵌套,回调显示那么苍白无力: async / await是ES7的重要特性之一,也是目前社区里公认的优秀异步解决方案,既然这样就用上吧. 二.配 ...
- ES7中的async和await
ES7中的async和await 在上一章中,使用Promise将原本的回调方式转换为链式操作,这就将一个个异步执行的操作串在一条同步线上了.下一次的操作必须等待当前操作的结束. 使用Promise的 ...
- 【TypeScript】如何在TypeScript中使用async/await,让你的代码更像C#。
[TypeScript]如何在TypeScript中使用async/await,让你的代码更像C#. async/await 提到这个东西,大家应该都很熟悉.最出名的可能就是C#中的,但也有其它语言也 ...
- 在现有代码中通过async/await实现并行
在现有代码中通过async/await实现并行 一项新技术或者一个新特性,只有你用它解决实际问题后,才能真正体会到它的魅力,真正理解它.也期待大家能够多分享解一些解决实际问题的内容. 在我们遭遇“黑色 ...
- js中的async await
JavaScript 中的 async/await 是属于比较新的知识,在ES7中被提案在列,然而我们强大的babel粑粑已经对它进行列支持! 如果开发中使用了babel转码,那么就放心大胆的用吧. ...
- [C#] .NET4.0中使用4.5中的 async/await 功能实现异
好东西需要分享 原文出自:http://www.itnose.net/detail/6091186.html 在.NET Framework 4.5中添加了新的异步操作库,但是在.NET Framew ...
- 在Silverlight中使用async/await
现在 async/await 大行其道,确实,有了 async/await ,异步编程真是简单多了,个人觉得 async/await 的出现,给开发者还来的方便,绝不亚于当年 linq 的出现. 但要 ...
随机推荐
- IDEA出现Cannot resolve symbol "xxx"(无法解析符号)
在导入一些包的时候出现报错 1.File->Invalidate Caches/Restart 清除缓存并重启 idea2.检查pom文件中的依赖关系是否正确3.maven -> Reim ...
- 大数据之 Spark
1 渊源 于2009由Matei Zaharia创立了spark大数据处理和计算框架,基于内存,用scala编写. 2 部署 2.1 需要软件包 下载路径见已有博文 Jdk ——因为运行环境为jvm ...
- Why is one loop so much slower than two loops?
Question: Suppose a1, b1, c1, and d1 point to heap memory and my numerical code has the following co ...
- Jmeter在非GUI(命令行)模式下生成测试报告
根据各大招聘网站上的需求来看,熟悉Jmeter做性能测试已经几乎成为必要条件了. 那么今天在这个给大家安利一波,怎么使用Jmeter在非GUI(命令行)模式下生成测试报告呢?? 条件准备: 1.Jme ...
- Android 动态渐变按钮
先上个图 看着特别炫酷吧 其实就是自定义颜色两秒轮播动画 AnimationDrawable animationDrawable = (AnimationDrawable) button.getBac ...
- Android 高德地图定位
创建Key 打开高德开发平台 → 我的应用 → 创建应用 → 创建新Key 说明: 1.发布版安全码获取:用自己的签名打包成apk安装软件,用SHA1工具查看 2.调试版安全码获取: 直接运行安装软件 ...
- maven——依赖管理
管理包依赖是 Maven 核心功能之一,下面通过如何引入 jar 包:如何解析 jar 包依赖:包冲突是如何产生:如何解决包冲突:依赖管理解决什么问题:什么是依赖范围:使用包依赖的最佳实践等 6 个问 ...
- python第三十一天-----类的封装、继承,多态.....
封装 封装最好理解了.封装是面向对象的特征之一,是对象和类概念的主要特性. 封装,也就是把客观事物封装成抽象的类,并且类可以把自己的数据和方法只让可信的类或者对象操作,对不可信的进行信息隐藏. cla ...
- Java同步、异步区别
一.概念: 1.同步:所有的操作都做完,才返回给用户.这样用户在线等待的时间太长,给用户一种卡死了的感觉(就是系统迁移中,点击了迁移,界面就不动了,但是程序还在执行,卡死了的感觉).这种情况下,用户不 ...
- plsql developer如何查询SQL语句执行历史记录(转)
相信很多在plsql developer调试oracle的朋友,经常会遇到在plsql developer执行的某一条SQL语句没有保存,那么我们在plsql developer下如何找到我们执行过的 ...