原文地址:http://javascriptplayground.com/blog/2015/02/promises/?utm_source=javascriptweekly&utm_medium=email
这篇文章我们一起来看下在异步编程事怎么拥抱promise,编写良好的代码。这篇不是全面深入地剖析Promise,如果想更全面的了解,Jake Archibald's post on HTML5 Rocks覆盖了方方面面。 强烈建议你阅读下。
尽管这篇文章中的代码依赖es-6-promise 库,它却是将要实现的ES6中一种补充。所有代码都通过运行nodejs执行,但是它应当有运行在浏览器环境的能力。当代码运行到Promise的时候,将会启用上面的es6-promise库,但是如果浏览器已经广泛地实现了promise特性,那么这些代码也可以同样的执行。
处理errors
第一个特性就是处理错误信息。这也是很多人都曾经问过的,很多人都遇到过的理解上陷阱。看下下面的代码,当执行这段代码的时候,你想会输出什么?
var someAsynThing = function(){ return new Promise(function(resolve,reject){ resolve(x +2 ); }); } someAsynThing().then(function(){ console.log('Everything is Ok'); });
你可能想的是抛出一个错误,因为变量X没有定义。如果你把代码写在promise外,就会发生你期望的情况。然而这块代码却相安无事,控制台什么都没有输出,并且也没有抛出任何错误。在promise内部,任何的错误都被和谐了,被理解为promise rejecting,这就意味着我们需要处理下error的情况。
someAsynThing().then(function(){ console.log('Everything is Ok'); }).catch(function(error){ console.log('oh no',error); });
运行下代码,
ReferenceError: x is not defined
你也可以通过使用promise链使errors看起来比较舒服,看下下面的实例代码
var someAsynThing = function(){ return new Promise(function(resolve,reject){ resolve(x +2 ); }); } var someOtherAsynThing = function(){ return new Promise(function(resolve,reject){ reject("Something went wrong"); }); } someAsynThing().then(function(){ return someOtherAsynThing(); }).catch(function(error){ console.log('oh no',error); });
我们依然看到同样的错误 oh no [ReferenceError: x is not defined],因为someAsynTing rejected,如果这个方法resolves,我们就会看到someOtherAsynThing reject(即“Something went wrong”)
var someAsynThing = function(){ return new Promise(function(resolve,reject){ var x =2; resolve(x +2 ); }); } var someOtherAsynThing = function(){ return new Promise(function(resolve,reject){ reject("Something went wrong"); }); } someAsynThing().then(function(){ return someOtherAsynThing(); }).catch(function(error){ console.log('oh no',error); });
现在我们看到的是oh no something went wrong.当一个promise rejects,promise链的第一个catch就会被调用。
另一个重要点就是没有特别的东西注册给catch。当一个promise rejects时只是一个一般的方法进行处理,而不会阻止其他的错作。
someAsynThing().then(function() { return someOtherAsynThing();}).catch(function(error) { console.log('oh no', error);}).then(function() { console.log('carry on');} );
一但rejects,“carry on”将会打印出来。当然catch内的handle会抛出一个错误,
someAsynThing().then(function() { return someOtherAsynThing(); }).catch(function(error) { console.log('oh no', error); y +2; }).then(function() { console.log('carry on'); });
这时catch中的回调方法被调用,但是不再显示“carry on”,因为catch中抛出一个错误。如果在promise链结束处增加一个catch方法,这个catch会继续运行,因为在一个catch中抛出,下一个catch将会调用。
链接和各种承诺传递
这部分是从我最近的工作受到的启发,导出CSV文件。做这个功能使用angularjs内置的$q服务,在这里做一个简单介绍。
导出CSV文件(CSV是浏览器内置的格式,借助FileSaver)的步骤
1、通过API接口读取数据来组装CSV(也可能是多个API请求)
2、将数据传递给CSV初始化对象
3、向CSV文件写入数据
4、文件创建成功或失败都要给用户一个确认消息
我们不会详细介绍代码怎么实现,只是从更高的层次的介绍下使用Promise构建健壮的解决方案。业务复杂的处理,错误随时都会发生(api加载,数据转换,或者CSV没有正确保存)。我们使用Promise的then和catch后才会发现这种处理方式更加的优雅。
正如你看到的,promise很笨重地结束了。但是我个人认为Promise链使得代码非常优美,但是刚开始你觉得有些古怪,Jake Archibald 能帮你的忙,让这种实现更好:
当从then回调中返回什么时候时,感觉很神奇。如果返回一个值,下一个then会被调用(这个值会被传递过去)。如果返回一个promise,下一个then将会等待,只有等前面的promise success/fails时,这个then中的回调才会被调用。
再次强调下,如果想更加深刻地了解promise,强烈建议你了解下http://www.html5rocks.com/en/tutorials/es6/promises/
现在我们开始实现一个简单的例子,返回写简单的数据。真实的application中可能是http请求或者别的类型。我们的promise讲resolve一个数组,将这些数据导出到CSV。
var fetchData = function() { return new Promise(function(resolve, reject) { setTimeout(function() { resolve({ users: [ { name: 'Jack', age: 22 }, { name: 'Tom', age: 21 }, { name: 'Isaac', age: 21 }, { name: 'Iain', age: 20 } ] }); }, 50); });}
接下来,需要一个function准备CSV 数据,立即resolve数据,但是在真实的application中需要做更多的work。
var prepareDataForCsv = function(data) { return new Promise(function(resolve, reject) { // imagine this did something with the data resolve(data); });};
这里我们还有很多注意点:prepareDataForCsv 不是异步的,并且没有必要把它封装到promise中。但是一个function作为操作链的一部分时,我们发现包装到一个promise内带来很大的好处,因为所有的error都可以通过promise处理。
最后,需要提供一个function向CSV写入数据,
var writeToCsv = function(data) { return new Promise(function(resolve, reject) { // write to CSV resolve(); });};
现在,我们将它们组合在一起。
fetchData().then(function(data) { return prepareDataForCsv(data);}).then(function(data) { return writeToCsv(data);}).then(function() { console.log('your csv has been saved');});
这种实现相当简洁,并且执行流程很清晰,我们还可以整理下更加紧凑。如果有一个方法带有一个参数,就可以将方法名直接传进去,而不是在回调方法中调用。
fetchData().then(prepareDataForCsv).then(writeToCsv).then(function() { console.log('your csv has been saved');});
底层的实现确实很是复杂,但是高层的API确实很好。我越来越欣赏promise,一但使用他们(complex code )并且成功运行,但是可以使用看起来比较优美的代码让这些复杂的实现结束。
然而,现在我们不需要error处理,但是我们可以增加一段额外代码。
fetchData().then(prepareDataForCsv).then(writeToCsv).then(function() { console.log('your csv has been saved');}).catch(function(error) { console.log('something went wrong', error);});
正是因为promise的作用链和error是怎么工作的,前面已经讨论,作用链的结束处的catch能够捕捉到任何的异常,使得error的处理径直向前。
为了说明这个,改造下prepareDataForCsv方法:
var prepareDataForCsv = function(data) { return new Promise(function(resolve, reject) { // imagine this did something with the data reject('data invalid'); });};
现在运行代码记录下错误。这是相当真棒 - prepareDataForCsv是正确的,在我们的promise 链的中间,但我们没有做任何额外的工作或弄虚作假处理错误。此外,catch不仅会捕捉我们通过promise rejects 的错误,但也抛出异常。这意味着,即使真是出乎意料边缘的情况下触发一个JS异常,用户仍然有自己的错误处理,符合预期结果。
我发现另一个非常强大的办法是期待一些数据,而不是使用promise resolve一些数据。让我们prepareDataForCsv为例:
var prepareDataForCsv = function(dataPromise) { return dataPromise().then(function(data) { return data; });};
我们发现这是一个相当不错的模式用来整理代码,并保持它更通用,其中大部分的工作是异步传递promise,而不是等着他们resolve和传送数据。
修改后的代码,如下:
prepareDataForCsv(fetchData).then(writeToCsv).then(function() { console.log('your csv has been saved');}).catch(function(error) { console.log('something went wrong', error);});
这样做的好处是,错误处理并没有改变。 fetchData可以reject某种形式,并且该错误将仍然在最后捕获处理。一旦你认同了这一点,你会发现promise非常好用,甚至更好的处理错误。
Promise递归
其中一个我们不得不面对的问题是,有时得从我们的API获取数据,可能需要进行多次请求。这是因为分页请求,所以如果数据量很大,你需要请求多次。值得庆幸的是我们的API告诉你,是否获取更多的数据,在此节中,我将解释我们是如何与Promise配合使用递归加载所有这些数据。
var count = 0;var http = function() { if(count === 0) { count++; return Promise.resolve({ more: true, user: { name: 'jack', age: 22 } }); } else { return Promise.resolve({ more: false, user: { name: 'isaac', age: 21 } }); }};
首先,我们有HTTP,这将作为假HTTP调用我们的API。 (Promise.resolve只是创建了一个promise,不管你是怎么调用都会resolve)。我第一次创建请求,将标志设置为true,这表明有更多的数据获取(这并不是现实中API怎么响应回应,但这篇文章的目的也是为了这个目的)。第二次该请求将more字段设置为false。因此,要获取所有需要的数据,我们需要两个API调用。让我们写一个函数fetchData来处理:
var fetchData = function() { var goFetch = function(users) { return http().then(function(data) { users.push(data.user); if(data.more) { return goFetch(users); } else { return users; } }); } return goFetch([]);};
fetchData本身确实非常短小,除了定义,然后调用另一个函数,goFetch。 goFetch返回一个用户数组(以goFetch初始话传递一个空数组),然后调用HTTP(),它resolve一组数据。返回的新的用户被存入用户数组,然后将功能着眼于data.more字段。如果这是真的,它再次调用自身,传递用户的新数组。如果它是假的,不在调用,它只是返回用户数组。这里最重要的事情,这么实现的原因是,在每一个阶段的东西被返回。 fetchData返回goFetch,它要么返回本身或用户数组。
结论
承诺不是随处可见,但是会成为处理大量异步操作的标准方法。然而,我发现其中的操作,有些是同步的,有些是异步的,一般提供了很多便利。如果你还没有尝试过但我真的建议在你的下一个项目试试。