data:image/s3,"s3://crabby-images/b5d20/b5d20dd1b25f20317f040a9fab5237c2c1ed7bae" alt="深入理解TypeScript"
3.13 async/await
请尝试想象以下场景:当使用Promise时,有一种方式告诉运行时的JavaScript暂停执行await关键字之后的内容,直到Promise从一个函数中执行完成。
data:image/s3,"s3://crabby-images/5dd6e/5dd6e870a9fc0902e7eaa0263592c52c3be63652" alt=""
当Promise完成时,将继续执行await关键字之后的代码。
● 如果成功完成,那么它会返回一个值。
● 如果它被拒绝,将同步抛出一个错误,我们可以用catch来捕获这个错误。
这突然且神奇地使异步编程变得像同步编程一样简单,这个实现需要3件事情。
● 能够停止函数执行。
● 能够在函数中放置一个值。
● 能够在函数内部抛出异常。
这正是generators允许我们做的事情,这个实验其实是真实的,async/await在JavaScript/TypeScript中正是这么实现的,它的背后只是使用了generators。
1.产生JavaScript
你不必理解这些,如果你已经阅读了generators就会发现它真的很简单。函数foo可以简单包装如下。
data:image/s3,"s3://crabby-images/67b99/67b9969b948cb81213b9922fd749bb778ba57878" alt=""
data:image/s3,"s3://crabby-images/9fafa/9fafa52f2ea5163f71f57dddc705472ea1d67901" alt=""
在上述代码中,wrapToReturnPromise执行generators函数来获取generators,接着使用generator.next(),如果值是Promise,则它会捕获Promise,并根据其结果调用generator.next(result)或generator.throw(error)。
2.TypeScript对async/await的支持
TypeScript从1.7版本之后开始支持async/await。异步函数以async关键字为前缀,await将暂停执行,直到异步函数返回Promise执行完成的结果,并且返回非Promise包装后的值。它只有在编译目标为ES6时支持async/await,并将其编译成ES6 generators。
TypeScript 2.1版本添加了ES3和ES5运行时的功能,这意味着无论你在何种环境下,都可以自由地使用它。值得注意的一点是,我们可以在TypeScript 2.1版本中使用async/await,许多浏览器都支持它。当然,你需要为Promise添加全局的polyfill。
下面让我们来看看TypeScript中的async/await是如何运行的,示例如下。
data:image/s3,"s3://crabby-images/5794e/5794ec538e358f70efd1c2e5c4f4e02cfdb19936" alt=""
data:image/s3,"s3://crabby-images/b32ef/b32ef1df08ce82dd0069409244d80ea62a5097b2" alt=""
1)编译到ES6(--target es6)
data:image/s3,"s3://crabby-images/8ec53/8ec534ea9d3b30729500dbe58c52ae7928ebfc9f" alt=""
data:image/s3,"s3://crabby-images/bdcea/bdcea05192f07b7dd071d6c5edbb4de4238555ec" alt=""
2)编译到ES5(--target es5)
data:image/s3,"s3://crabby-images/78631/78631a40e57fa611dff61ecb034093faf19e7445" alt=""
data:image/s3,"s3://crabby-images/e2c11/e2c11b90629299bd7365982854c31b9e573df0af" alt=""
data:image/s3,"s3://crabby-images/cbdd1/cbdd1f16ffda9ab5de220601331ac847ff1c213b" alt=""
data:image/s3,"s3://crabby-images/0483d/0483da3985cf5089b2da6bd0f6e1068710ac2e39" alt=""
data:image/s3,"s3://crabby-images/ee62a/ee62a01fe9fd16452142e90c86bbfeb3b5c4e001" alt=""
data:image/s3,"s3://crabby-images/bbe84/bbe8469e5704c83130078342d73e83e27e61a60f" alt=""
完整的示例见参考资料[5]。
注意:对于这种有两种编译目标的场景,我们都需要确保运行时在全局下有一个符合ECMAScript规范的Promise,它可能是一个Promise的polyfill。同样,我们也需要确定TypeScript知道Promise的存在,这可以通过将lib标记设置为dom、es2015、dom、es2015.promise、es5来实现。在参考资料[6]中,可以看到支持Promise(包括native和polyfill)的浏览器都有哪些。