本文内容转载于 边城 博客
原文链接 https://segmentfault.com/a/1190000007535316
理解 JavaScript 的 async/await
async 和 await 在干什么
任意一个名称都是有意义的,先从字面意思来理解。async
是“异步”的简写,而 await
可以认为是 async wait
的简写。所以应该很好理解 async
用于申明一个 function
是异步的,而 await
用于等待一个异步方法执行完成。
另外还有一个很有意思的语法规定,await
只能出现在 async
函数中。然后细心的朋友会产生一个疑问,如果 await
只能出现在 async
函数中,那这个 async
函数应该怎么调用?
如果需要通过 await
来调用一个 async
函数,那这个调用的外面必须得再包一个 async
函数,然后……进入死循环,永无出头之日……
如果 async
函数不需要 await
来调用,那 async
到底起个啥作用?
async 起什么作用
这个问题的关键在于,async
函数是怎么处理它的返回值的!
我们当然希望它能直接通过 return
语句返回我们想要的值,但是如果真是这样,似乎就没 await 什么事了。所以,写段代码来试试,看它到底会返回什么:
async function testAsync() {
return "hello async";
}
const result = testAsync();
console.log(result);
看到输出就恍然大悟了——输出的是一个 Promise 对象。
c:\var\test> node --harmony_async_await .
Promise { 'hello async' }
所以,async
函数返回的是一个 Promise
对象。从文档中也可以得到这个信息。async
函数(包含函数语句、函数表达式、Lambda表达式)会返回一个 Promise
对象,如果在函数中 return
一个直接量,async
会把这个直接量通过 Promise.resolve()
封装成 Promise
对象。
async
函数返回的是一个 Promise
对象,所以在最外层不能用 await
获取其返回值的情况下,我们当然应该用原来的方式:then()
链来处理这个 Promise
对象,就像这样
testAsync().then(v => {
console.log(v); // 输出 hello async
});
现在回过头来想下,如果 async
函数没有返回值,又该如何?很容易想到,它会返回
Promise.resolve(undefined)
联想一下 Promise
的特点——无等待,所以在没有 await 的情况下执行 async
函数,它会立即执行,返回一个 Promise
对象,并且,绝不会阻塞后面的语句。这和普通返回 Promise
对象的函数并无二致。
那么下一个关键点就在于 await
关键字了。
await 到底在等啥
一般来说,都认为 await
是在等待一个 async
函数完成。不过按语法说明,await
等待的是一个表达式,这个表达式的计算结果是 Promise
对象或者其它值(换句话说,就是没有特殊限定)。
因为 async 函数返回一个 Promise
对象,所以 await
可以用于等待一个 async
函数的返回值——这也可以说是 await
在等 async
函数,但要清楚,它等的实际是一个返回值。注意到 await 不仅仅用于等 Promise
对象,它可以等任意表达式的结果,所以,await
后面实际是可以接普通函数调用或者直接量的。所以下面这个示例完全可以正确运行
function getSomething() {
return "something";
}
async function testAsync() {
return Promise.resolve("hello async");
}
async function test() {
const v1 = await getSomething();
const v2 = await testAsync();
console.log(v1, v2);
}
test();
await 等到了要等的,然后呢
await
等到了它要等的东西,一个 Promise
对象,或者其它值,然后呢?我不得不先说,await
是个运算符,用于组成表达式,await
表达式的运算结果取决于它等的东西。
如果它等到的不是一个 Promise
对象,那 await
表达式的运算结果就是它等到的东西。
如果它等到的是一个 Promise
对象,await
就忙起来了,它会阻塞后面的代码,等着 Promise
对象 resolve
,然后得到 resolve
的值,作为 await
表达式的运算结果。
看到上面的阻塞一词,心慌了吧……放心,这就是 await 必须用在 async
函数中的原因。async
函数调用不会造成阻塞,它内部所有的阻塞都被封装在一个Promise
对象中异步执行。
async/await 帮我们干了啥
作个简单的比较
上面已经说明了 async
会将其后的函数(函数表达式或 Lambda)的返回值封装成一个 Promise 对象,而 await 会等待这个 Promise
完成,并将其resolve
的结果返回出来。
现在举例,用 setTimeout
模拟耗时的异步操作,先来看看不用 async/await 会怎么写
function takeLongTime() {
return new Promise(resolve => {
setTimeout(() => resolve("long_time_value"), 1000);
});
}
takeLongTime().then(v => {
console.log("got", v);
});
如果改用 async/await 呢,会是这样
```bash
function takeLongTime() {
return new Promise(resolve => {
setTimeout(() => resolve("long_time_value"), 1000);
});
}
async function test() {
const v = await takeLongTime();
console.log(v);
}
test();
眼尖的同学已经发现takeLongTime()
没有申明为 async
。实际上,takeLongTime()
本身就是返回的 Promise
对象,加不加 async 结果都一样,如果没明白,请回过头再去看看上面的async 起什么作用
。
又一个疑问产生了,这两段代码,两种方式对异步调用的处理(实际就是对 Promise 对象的处理)差别并不明显,甚至使用 async/await 还需要多写一些代码,那它的优势到底在哪?
async/await 的优势在于处理 then 链
单一的 Promise 链并不能发现 async/await
的优势,但是,如果需要处理由多个 Promise 组成的 then 链的时候,优势就能体现出来了(很有意思,Promise 通过 then 链来解决多层回调的问题,现在又用 async/await
来进一步优化它)。
假设一个业务,分多个步骤完成,每个步骤都是异步的,而且依赖于上一个步骤的结果。我们仍然用 setTimeout
来模拟异步操作:
/**
* 传入参数 n,表示这个函数执行的时间(毫秒)
* 执行的结果是 n + 200,这个值将用于下一步骤
*/
function takeLongTime(n) {
return new Promise(resolve => {
setTimeout(() => resolve(n + 200), n);
});
}
function step1(n) {
console.log(`step1 with ${n}`);
return takeLongTime(n);
}
function step2(n) {
console.log(`step2 with ${n}`);
return takeLongTime(n);
}
function step3(n) {
console.log(`step3 with ${n}`);
return takeLongTime(n);
}
现在用 Promise 方式来实现这三个步骤的处理
function doIt() {
console.time("doIt");
const time1 = 300;
step1(time1)
.then(time2 => step2(time2))
.then(time3 => step3(time3))
.then(result => {
console.log(`result is ${result}`);
console.timeEnd("doIt");
});
}
doIt();
// c:\var\test>node --harmony_async_await .
// step1 with 300
// step2 with 500
// step3 with 700
// result is 900
// doIt: 1507.251ms
输出结果 result 是 step3() 的参数 700 + 200 = 900
。doIt() 顺序执行了三个步骤,一共用了 300 + 500 + 700 = 1500
毫秒,和 console.time()/console.timeEnd()
计算的结果一致。
如果用 async/await 来实现呢,会是这样
async function doIt() {
console.time("doIt");
const time1 = 300;
const time2 = await step1(time1);
const time3 = await step2(time2);
const result = await step3(time3);
console.log(`result is ${result}`);
console.timeEnd("doIt");
}
doIt();
结果和之前的 Promise
实现是一样的,但是这个代码看起来是不是清晰得多,几乎跟同步代码一样
还有更酷的
现在把业务要求改一下,仍然是三个步骤,但每一个步骤都需要之前每个步骤的结果。
function step1(n) {
console.log(`step1 with ${n}`);
return takeLongTime(n);
}
function step2(m, n) {
console.log(`step2 with ${m} and ${n}`);
return takeLongTime(m + n);
}
function step3(k, m, n) {
console.log(`step3 with ${k}, ${m} and ${n}`);
return takeLongTime(k + m + n);
}
这回先用 async/await 来写:
async function doIt() {
console.time("doIt");
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}`);
console.timeEnd("doIt");
}
doIt();
// c:\var\test>node --harmony_async_await .
// step1 with 300
// step2 with 800 = 300 + 500
// step3 with 1800 = 300 + 500 + 1000
// result is 2000
// doIt: 2907.387ms
除了觉得执行时间变长了之外,似乎和之前的示例没啥区别啊!别急,认真想想如果把它写成 Promise
方式实现会是什么样子?
function doit() {
console.time('doit');
const time1 = 300;
step1(time1)
.then(time2 => {
return step2(time1, time2)
.then(time3 => {
return step3(time1, time2, time3)
.then(result => {
console.log(`result is ${result}`)
console.timeEnd('doit')
})
})
})
}
doIt();
//可以换种写法
function doit() {
console.time('doit');
var time1 = 300, time2, time3;
step1(time1)
.then(param => step2(time1, time2 = param))
.then(param => step3(time1, time2, time3 = param))
.then(result => {
console.log(`result is ${result}`);
console.timeEnd('doit')
})
}
doIt();
有没有感觉有点复杂的样子?那一堆参数处理,就是Promise
方案的死穴—— 参数传递太麻烦了,看着就晕!
本文内容转载于 边城 博客
原文链接 https://segmentfault.com/a/1190000007535316