data:image/s3,"s3://crabby-images/b5d20/b5d20dd1b25f20317f040a9fab5237c2c1ed7bae" alt="深入理解TypeScript"
3.11 Promise
Promise类存在于许多现代的JavaScript引擎中,可以轻松地被polyfill。Promise的主要目的是用同步的代码替换异步/回调函数的代码。
1.回调函数风格的代码
为了完全理解Promise,我们将提供一个简单的示例来证明使用回调函数创建可靠的异步代码——从一个文件中异步加载JSON文件是非常困难的,一个同步操作的版本如下。
data:image/s3,"s3://crabby-images/da502/da502d7033b843c6bfce991f9c4b0e632c742c51" alt=""
data:image/s3,"s3://crabby-images/84de4/84de45464196cf438673e3ba9a9a9a76986f4cef" alt=""
一个简单的loadJSONSync函数会有如下结果:一个有效的返回值、一个文件系统错误或一个JSON.parse错误。我们使用try/catch来捕获错误,就像你在其他语言中处理同步中的错误所习惯的那样。现在,让我们来为这个函数写一个好的异步版本吧,使用简单错误检查逻辑是一个不错的开始。
data:image/s3,"s3://crabby-images/ce654/ce654357f3d8f82cb610f4b8ab8b11bb07364c4c" alt=""
很简单,它需要一个回调函数,并将文件系统错误传递给该函数;如果没有文件系统错误,它会返回JSON.parse的结果。在使用基于回调的异步函数时,要记住以下两点。
● 永远不要调用两次回调函数。
● 永远不要抛出错误。
但是,这个简单的函数却不符合第2点。实际上,当JSON.parse被传入错误的JSON文件,并且回调函数一直未被调用而导致程序崩溃时,JSON.parse会抛出错误,如下所示。
data:image/s3,"s3://crabby-images/963a0/963a036448545fdab6b7e84deb42ed59b01b037a" alt=""
解决这个问题的一个尝试,是将JSON.parse包装在try/catch中,如下所示。
data:image/s3,"s3://crabby-images/a6a21/a6a21159295b7506acc3ba279cae10e8e518de7a" alt=""
data:image/s3,"s3://crabby-images/2da01/2da013fee4a44b31f451a15e9c1b558232f6b27f" alt=""
但是,此代码有一个小错误,如果是回调函数(cb),而不是JSON.parse,会抛出一个错误。因为我们将它包装在try/catch中,当catch执行时,我们将会再次执行回调函数,即回调函数执行了两次,如下所示。
data:image/s3,"s3://crabby-images/629cd/629cdfa4bef8b93fef7765aecefbcca89b21170d" alt=""
data:image/s3,"s3://crabby-images/aed16/aed1688066585c61fe186bf5144b6f10e948fa19" alt=""
出现这种现象的原因是loadJSON函数错误地把一个回调函数包装在了一个try块内。
注意:建议把所有的同步代码都放入try/catch,除了回调函数。
按照这个提示,我们将会有一个功能齐全的loadJSON异步版本,如下所示。
data:image/s3,"s3://crabby-images/9e16c/9e16cbb6c15f7e85ac6dfeb4fd77505852554ea1" alt=""
data:image/s3,"s3://crabby-images/65b44/65b4488261b669de023e73eda1134125b59f5a17" alt=""
诚然,在尝试几次之后,这将并不难理解。但尽管如此,为了良好地处理错误,仍需要编写大量的重复代码。现在,让我们看一个更好的方法:在JavaScript中使用Promise处理异步。
2.使用Promise处理异步和回调函数
1)创建一个Promise
一个Promise可以处于pending、fulfilled或rejected状态。
data:image/s3,"s3://crabby-images/dbbd6/dbbd6b5d803f2a52bb67e1f95e2757f1b4d93b80" alt=""
让我们来创建一个Promise,在Promise(Promise的构造函数)上调用new是一件非常容易的事情。Promise构造函数通过resolve和reject函数设置Promise的状态。
data:image/s3,"s3://crabby-images/ff257/ff257dff53de9ff4582a6fc113d79df80d9bd9c2" alt=""
2)订阅Promise的状态
Promise的状态能通过then函数(如果resolved)和catch函数(如果reject)被订阅。
data:image/s3,"s3://crabby-images/3526e/3526e5b9eab8c8dd0149e8a9a2bb4dcbf49bf36d" alt=""
提示:创建Promise的快捷方式如下。
● 快速创建已经resolved的Promise:Promise.resolve(result)。
● 快速创建已经rejected的Promise:Promise.reject(error)。
3)Promise的链式调用
Promise的链式调用是Promise提供的核心价值,一旦有了一个Promise,你就可以使用then函数来创建链式的Promise了。
● 如果从链中的任何一个函数返回一个Promise,当值是resolved时,只有.then会被调用。
data:image/s3,"s3://crabby-images/d8555/d8555b4fa42f60c1d97a6f8985e374f9f3d72bae" alt=""
● 你可以使用单个.catch来捕获前面的链中抛出的异常。
data:image/s3,"s3://crabby-images/72d65/72d65b659b5c8714b1bb0c2a8d02eca00882f7c2" alt=""
● catch实际上返回了一个新的Promise(有效地创建了一个新的Promise)。
data:image/s3,"s3://crabby-images/f635b/f635b76383e31e6dbf03c1160927f5cf62fd0983" alt=""
● 在then(或catch)中抛出的任何同步错误都会导致返回的Promise失败。
data:image/s3,"s3://crabby-images/71e90/71e90b617e3d340b2d85192231b69352b5647751" alt=""
● 给定的错误(当catch启用一个新的Promise时)只调用相关的(最靠近尾部的)catch。
data:image/s3,"s3://crabby-images/38bac/38bacd35d93d0d97a5781ec1e59d6cb54bfe1a84" alt=""
data:image/s3,"s3://crabby-images/da763/da76388ab04cec6b76fbd9a32cd5b7b53a745b24" alt=""
● 只有在前面的链中抛出错误时,才会调用catch。
data:image/s3,"s3://crabby-images/1faca/1faca6aecea47c8f8698ad854a29e01d1295d285" alt=""
事实是抛出的错误会直接进入尾部的catch(跳过中间的then);同步的错误也会被任何一个尾部的catch捕获到。
Promise有效地为我们提供了一个异步编程范例,比起使用回调函数,它能让我们更好地处理错误。
4)TypeScript和Promise
TypeScript的优点在于它通过Promise链能理解从Promise传过来的值。
data:image/s3,"s3://crabby-images/8def3/8def3b2e1814061f1ca904e728d9013bb2cfe27a" alt=""
data:image/s3,"s3://crabby-images/e8bfc/e8bfc72d8f3a5b673ada691f64c98193d802d2df" alt=""
当然,它也能理解返回值可能为Promise的任何非包装函数。
data:image/s3,"s3://crabby-images/08116/08116ddf1eb39542ae83ed57e5895d7eff634b74" alt=""
5)转换回调风格的函数以返回Promise
只需把函数的调用包裹在一个Promise中,并在有错误发生时,使用reject;如果一切正常,则使用resolve。
例如,包裹fs.readFile。
data:image/s3,"s3://crabby-images/227aa/227aa93b99c966fa6269b0158b8231b66b089eea" alt=""
data:image/s3,"s3://crabby-images/e249c/e249c820c8802cc393f181d06e4888f579927bcd" alt=""
最可靠的方式是手写代码来实现它,这样会非常简单,不会像上文的实例那样冗长,例如将setTimeout转化为一个Promise的delay函数。
data:image/s3,"s3://crabby-images/8134b/8134beb9f776c4b732383bc940808d5c06dadd94" alt=""
注意:在Node.js中已经有一个花哨的函数了,它可以为你提供这样的node style function=>promise returning function魔法。
data:image/s3,"s3://crabby-images/dbf62/dbf62eab3da883d4b0df5cabc37763dbaf6a9ce3" alt=""
webpack能够“开箱即用”util模块,你也可以在浏览器中使用它。
如果你有一个Node回调风格的函数作为成员,请使用bind来确保拥有正确的this。
6)重温JSON的例子
现在让我们重新回顾loadJSON这个例子,并使用Promise的异步版本重写它。我们所需要做的事情是使用Promise读取文件内容,然后将它们解析为JSON文件,示例如下。
data:image/s3,"s3://crabby-images/1f468/1f468221616cc083fa4dcce047d34816e0fb701f" alt=""
用法如下(注意它与本节开头介绍的同步版本是相似的)。
data:image/s3,"s3://crabby-images/86226/862268d38ca42c534a56e9073f086556dcb72f83" alt=""
data:image/s3,"s3://crabby-images/8afa6/8afa609f28d7502ebd4eb6447fcfde95b3615c9e" alt=""
这个函数如此简单的原因是,"loadFile(async)+JSON.parse(sync)=>catch"由Promise链式合并完成。同样,这个回调函数也没有派上用场,它由Promise链式调用完成。因此我们无须将它包裹在try/catch中。
3.并行流程控制
我们已经看到了使用Promise来写异步任务是多么简单,这只是一个简单的链式调用then的问题。
然而你可能希望并行运行一系列异步任务,然后处理所有任务的结果。Promise提供了一个静态的Promise.all函数,你可以使用该函数等待n个Promise完成。如果你提供一个含有n个Promise的数组,它将返回一个包含n个已resolved的值的数组。示例如下。
data:image/s3,"s3://crabby-images/65ad1/65ad15b0659b84621b2df0439ce4373e30256c2c" alt=""
data:image/s3,"s3://crabby-images/315f0/315f08ae0561bbf8894e02819ab6416fafbb9881" alt=""
有时,你可能希望运行多个异步任务,但这些任务只要有一个settled就可以了。Promise提供一个静态的Promise.race函数来处理这种情况。
data:image/s3,"s3://crabby-images/c568e/c568ea47839cc7e2272e3b1dd8cf88a29f738dd3" alt=""
4.将回调函数转为Promise
将回调函数转为Promise,最可靠的方法是手写,例如将setTimeout转换为Promise形式的delay函数。
data:image/s3,"s3://crabby-images/781cc/781ccc11edf7585801b769b953ae5440d687ff80" alt=""
请注意,Node.js中有一个便利的包装函数,它可以给你node style function=>promise returning function的技巧。
data:image/s3,"s3://crabby-images/e13ba/e13ba6d10d25aef4423a0f852bd3e8f3ebc31e46" alt=""