JavaScript 通过其流行度和最近的改进,日益成为 Web 程序员最好的朋友。和所有最好的朋友一样,JavaScript 信守承诺。
这听起来可能有点奇怪,但这是事实。当前大多数浏览器都支持所谓的 Promise 对象。 Promise 很像一个函数,它代表您希望在未来某个时刻执行的一段代码或一项任务。
这是承诺的样子。
var myPromise = new Promise(function (resolve, reject) { // Task to carry out goes here. });
您可以在这里看到,当我们创建一个 Promise 时,我们给它一个参数,这是一个包含我们希望在将来某个时刻执行的代码的函数。您可能还注意到函数中传递给 Promise 的两个参数: resolve 和 reject。这些也是函数,是我们告诉 Promise 是否完成了承诺的方式。这是您使用它们的方式:
var myPromise = new Promise(function (resolve, reject) { if (true) { resolve('Hello Tuts+ fans!'); } else { reject('Aww, didn't work.'); } });
这个承诺显然总是会得到解决,因为 if 语句将始终为真。这只是出于学习目的 – 我们稍后会做一些更实际的事情 – 但想象一下用一段您不能 100% 确定是否有效的代码片段替换 true 。
现在我们已经创建了一个 Promise,我们该如何使用它呢?好吧,我们需要告诉它 resolve 和 reject 函数是什么。我们通过使用 Promise 的 then 方法来做到这一点。
myPromise.then(function (result) { // Resolve callback. console.log(result); }, function (result) { // Reject callback. console.error(result); });
因为我们的 if 语句总是通过其 true 检查,所以上面的代码将始终记录“Hello Tuts+球迷!”到控制台。它也会立即执行。这是因为 Promise 构造函数内的代码是同步的,这意味着它不会等待任何操作的执行。它拥有继续进行所需的所有信息,并会尽快进行。
然而,Promise 真正发挥作用的地方是异步任务,即您不知道 Promise 何时具体实现的任务。异步任务的一个现实示例是通过 AJAX 获取资源,例如 JSON 文件。我们不知道服务器需要多长时间才能响应,甚至可能会失败。让我们在 Promise 代码中添加一些 AJAX。
var myPromise = new Promise(function (resolve, reject) { // Standard AJAX request setup and load. var request = new XMLHttpRequest(); // Request a user's comment from our fake blog. request.open('GET', 'http://jsonplaceholder.typicode.com/posts/1'); // Set function to call when resource is loaded. request.onload = function () { if (request.status === 200) { resolve(request.response); } else { reject('Page loaded, but status not OK.'); } }; // Set function to call when loading fails. request.onerror = function () { reject('Aww, didn't work at all.'); } request.send(); });
这里的代码只是用于执行 AJAX 请求的标准 JavaScript。我们请求一个资源,在本例中是指定 URL 处的 JSON 文件,并等待它响应。我们永远不会知道确切的时间。我们显然不想停止执行脚本来等待它,那么我们该怎么办呢?
幸运的是,我们已将此代码放入 Promise 中。把它放在这里,我们基本上是在说,“嘿,一段代码,我现在得走了,但我稍后会给你打电话,告诉你何时执行。保证你会这么做并告诉我你什么时候完成?”代码会说:“是的,当然。我保证。”
上面代码中需要注意的重要一点是 resolve 和 reject 函数的调用。请记住,这些是我们告诉我们的承诺我们的代码是否成功执行的方式。否则,我们永远不会知道。
使用基本示例中的相同代码,我们可以看到 Promise 中的 AJAX 请求现在将如何工作。
// Tell our promise to execute its code // and tell us when it's done. myPromise.then(function (result) { // Prints received JSON to the console. console.log(result); }, function (result) { // Prints "Aww didn't work" or // "Page loaded, but status not OK." console.error(result); });
我知道我们可以信任你,myPromise。
链接承诺
现在,您可能会认为 Promise 只是具有更好语法的奇特回调函数。这在某种程度上是正确的,但为了继续我们的 AJAX 示例,假设您需要再发出一些请求,每个请求都基于上一个请求的结果。或者如果您需要先处理 JSON 怎么办?
使用回调执行此操作将导致函数的大量嵌套,每个函数都变得越来越难以跟踪。幸运的是,在 Promise 的世界中,我们可以像这样将这些函数链接在一起。下面是一个示例,一旦我们收到用户在我们的假博客上发表的评论的 JSON,我们就需要确保它全部是小写,然后再对其进行其他操作。
myPromise .then(function (result) { // Once we receive JSON, // turn it into a JSON object and return. return JSON.parse(result); }) .then(function (parsedJSON) { // Once json has been parsed, // get the email address and make it lowercase. return parsedJSON.email.toLowerCase(); }) .then(function (emailAddress) { // Once text has been made lowercase, // print it to the console. console.log(emailAddress); }, function (err) { // Something in the above chain went wrong? // Print reject output. console.error(err); });
您可以在此处看到,虽然我们的初始调用是异步的,但也可以链接同步调用。 resolve 函数中的代码在 then 每个返回时都会被调用。您还会注意到,这里只为整个链指定了一个误差函数。通过将其作为最后一个 then 中的 reject 函数放在链的末尾,链中调用 reject 的任何 Promise 都将调用此函数一个。
现在我们对承诺更有信心了,让我们结合上面的承诺创建另一个承诺。我们将创建一个采用新的小写电子邮件地址的电子邮件地址,并(假装)向该地址发送电子邮件。这只是一个说明异步内容的示例 – 它可以是任何内容,例如联系服务器以查看电子邮件是否在白名单上或者用户是否已登录。我们需要将电子邮件地址提供给新的 Promise,但承诺不接受争论。解决这个问题的方法是将 Promise 包装在一个可以执行此操作的函数中,如下所示:
var sendEmail = function (emailAddress) { return new Promise(function (resolve, reject) { // Pretend to send an email // or do something else asynchronous setTimeout(function () { resolve('Email sent to ' + emailAddress); }, 3000); }); };
我们在此处使用 setTimeout 调用来简单地伪造一个需要几秒钟才能异步运行的任务。
那么我们如何使用新的承诺创建函数呢?好吧,由于在 then 中使用的每个 resolve 函数都应该返回一个函数,那么我们可以以与同步任务类似的方式使用它。
myPromise .then(function (result) { return JSON.parse(result); }) .then(function (parsedJSON) { return parsedJSON.email.toLowerCase(); }) .then(function (emailAddress) { return sendEmail(emailAddress) }) .then(function (result) { // Outputs "Email sent to stuart@fakemail.biz" console.log(result); }, function (err) { console.error(err); });
让我们回顾一下这个流程,总结一下发生了什么。我们最初的 Promise myPromise 请求一段 JSON。当收到该 JSON 时(我们不知道何时),我们将 JSON 转换为 JavaScript 对象并返回该值。
完成后,我们从 JSON 中取出电子邮件地址并将其变为小写。然后我们向该地址发送一封电子邮件,同样我们不知道它何时完成,但当它完成时,我们将向控制台输出一条成功消息。看不到沉重的嵌套。
结论
我希望这是对 Promise 的有用介绍,并为您提供了在 JavaScript 项目中开始使用它们的充分理由。如果您想更详细地了解 Promise,请查看 Jake Archibald 关于这个主题的优秀 HTML5 Rocks 文章。
学习 JavaScript:完整指南
我们构建了一个完整的指南来帮助您学习 JavaScript,无论您是刚刚开始作为 Web 开发人员还是想探索更高级的主题。
本站部分资源来源于网络,仅限用于学习和研究目的,请勿用于其他用途。
如有侵权请发送邮件至1943759704@qq.com删除
码农资源网 » JavaScript:恪守我们的承诺