英文原文: http://emberjs.com/guides/routing/asynchronous-routing/

本节内容主要介绍一些路由的高级特性,以及路由是如何处理应用中的一些复杂异步逻辑的。

承诺简介

Ember在路由中处理异步逻辑的方案主要依赖于承诺(Promise)。简单地说,承诺就是代表了最后的值的对象。承诺可以被履行(成功的获得了最后的结果)也可以被拒绝(没有获得最后的结果)。处理获取最后的值,或者承诺被拒绝的方法是通过承诺的then方法来实现的。该方法接收两个可选的回调函数,一个用于履行承诺时执行,一个则是在拒绝时执行。如果一个承诺被履行,那么最后的值会作为履行处理函数的参数传入;如果一个承诺被拒绝,拒绝的原因会作为拒绝处理函数的参数传入。例如:

  1. var promise = fetchTheAnswer();
  2. promise.then(fulfill, reject);
  3. function fulfill(answer) {
  4. console.log("The answer is " + answer);
  5. }
  6. function reject(reason) {
  7. console.log("Couldn't get the answer! Reason: " + reason);
  8. }

承诺的强大很大程度来源于承诺可以连接在一起,形成一个有序的异步操作链:

  1. // Note: jQuery AJAX methods return promises
  2. var usernamesPromise = Ember.$.getJSON('/usernames.json');
  3. usernamesPromise.then(fetchPhotosOfUsers)
  4. .then(applyInstagramFilters)
  5. .then(uploadTrendyPhotoAlbum)
  6. .then(displaySuccessMessage, handleErrors);

在上述的例子中,如果fetchPhotosOfUsersapplyInstagramFilters、或uploadTrendyPhotoAlbum中任意一个方法返回的承诺被拒绝了,handleErrors将会被调用,并且告知拒绝的原因。在这种方式下,承诺将异步行为组织为与try-catch相似的形式,避免了一层层向右嵌套的回调,提供了一种健全有效的管理应用中异步逻辑的方案。

本指南中并不涉及承诺工作的各种场景,如果希望知道更多的关于承诺的介绍,可以查看Ember使用的RSVP的README文件。

路由会为承诺暂停

当在路由间过渡时,Ember的路由通过model钩子获取所有将会在过渡的最后传递给路由控制器的模型。如果model钩子(或相关的beforeModelafterModel钩子)返回的是普通的对象或数组(非承诺),那么过渡会立即完成。但是如果model钩子(包括beforeModelafterModel钩子)返回一个承诺(或者将一个承诺作为参数传递给transitionTo方法),那么过渡将暂停直到承诺被履行或者拒绝。

如果承诺得到履行,那么过渡将从其停止的地方开始,并着手处理下一个路由(子路由)的模型,如果这个路由依然返回一个承诺,那么又将暂停进行等待,以此类推,知道所有目标路由的模型都被获取到了。传递给每个路由的setupController钩子的值都是履行完承诺后的值。

一个基本的示例:

  1. App.TardyRoute = Ember.Route.extend({
  2. model: function() {
  3. return new Ember.RSVP.Promise(function(resolve) {
  4. Ember.run.later(function() {
  5. resolve({ msg: "Hold Your Horses" });
  6. }, 3000);
  7. });
  8. },
  9. setupController: function(controller, model) {
  10. console.log(model.msg); // "Hold Your Horses"
  11. }
  12. });

当过渡到TardyRoute时,路由的model钩子将被调用,并且返回一个要三秒后才会被履行的承诺,在此期间路由会被暂停在一个中间状态。当承诺最后被履行时,路由将继续过渡并最后将获得的对象传递给TardyRoutesetupController钩子。

当在显示一个新模板之前需要确保路由的数据被完全加载成功的情况下,这种暂停在承诺的行为非常有用。

当承诺被拒绝时

之前讲到了一个承诺被履行的情况会是什么样,那么如果承诺被拒绝了又会是怎样呢?

在默认的情况下,如果一个模型承诺在过渡过程中被拒绝了,那么过渡就会被取消,不会有新目标路由的模板被渲染,控制台也会同时记录一条错误。

通过在路由的actions哈希里,定义一个error处理器,可以用来处理承诺被拒绝的情况。当一个承诺被拒绝时,路由上会被触发一个error事件,如果没有自定义的错误处理,那么该事件会一直冒泡直至ApplicationRoute提供的缺省错误处理器。

  1. App.GoodForNothingRoute = Ember.Route.extend({
  2. model: function() {
  3. return Ember.RSVP.reject("FAIL");
  4. },
  5. actions: {
  6. error: function(reason) {
  7. alert(reason); // "FAIL"
  8. // Can transition to another route here, e.g.
  9. // this.transitionTo('index');
  10. // Uncomment the line below to bubble this error event:
  11. // return true;
  12. }
  13. }
  14. });

在上述例子中,错误事件将停留在GoodForNothingRoute的错误处理器中,而不会往上冒泡。如果需要继续让事件向上冒泡到ApplicationRoute,那么需要在错误处理器里面返回true

从拒绝处恢复

被拒绝的模型承诺会停止过渡,但是由于承诺是可以链式相连的,那么就可以在model钩子里面捕获到承诺的拒绝,然后将其转换为履行,这样就不会停止过渡了。

  1. App.FunkyRoute = Ember.Route.extend({
  2. model: function() {
  3. return iHopeThisWorks().then(null, function() {
  4. // Promise rejected, fulfill with some default value to
  5. // use as the route's model and continue on with the transition
  6. return { msg: "Recovered from rejected promise" };
  7. });
  8. }
  9. });

beforeModelafterModel

model钩子包含了基于承诺暂停过渡的许多应用场景,但是有的时候还是需要beforeModelafterModel这两个钩子来提供帮助。最常见的原因是通过{{link-to}}或者{{transitionTo}}(最为URL改变导致的过渡的对比)过渡到一个具有动态URL端,这时将过渡到的路由的模型早早就被指定了(例如{{#link-to 'article' article}}或者this.transitionTo('article', article)),这种情况下,model钩子并不会被调用。当路由正在收集所有路由的模型来执行过渡时,就需要使用beforeModelafterModel钩子来处理所有逻辑。

beforeModel

beforeModel钩子是这两个中更为常用到的一个,beforeModel在路由器尝试解决给定路由的模型时被调用。或者说,它是在model钩子被调用前调用的。如果model钩子不会被调用,那么beforeModel会在路由器尝试解决传入到路由的模型承诺之前被调用。

model一样,如果beforeModel返回一个承诺,那么路由将会被暂停至承诺被履行时,或者是在承诺被拒绝时触发一个error事件。

下面详细的列出了beforeModel非常有用的几种用例:

  • 决定是否在执行一个潜在的在model中的费时的服务器端查询之前跳转到其他的路由
  • 确定用户在执行model之前有一个身份令牌
  • 加载路由所需要的应用代码
  1. App.SecretArticlesRoute = Ember.Route.extend({
  2. beforeModel: function() {
  3. if (!this.controllerFor('auth').get('isLoggedIn')) {
  4. this.transitionTo('login');
  5. }
  6. }
  7. });

查看beforeModelAPI文档

afterModel

afterModel钩子在路由的模型(可能是一个承诺)被解决之后调用,并与modelbeforeModel一样,遵从基于承诺的暂停原则。afterModel将传入被解决的模型,并可以在这个已经完全解决了的模型上添加一些附件的逻辑操作。

  1. App.ArticlesRoute = Ember.Route.extend({
  2. model: function() {
  3. // App.Article.find() returns a promise-like object
  4. // (it has a `then` method that can be used like a promise)
  5. return App.Article.find();
  6. },
  7. afterModel: function(articles) {
  8. if (articles.get('length') === 1) {
  9. this.transitionTo('article.show', articles.get('firstObject'));
  10. }
  11. }
  12. });

这里也许对为什么不将afterModel的逻辑放入model承诺履行时要执行的处理器而感到好奇;其实原因很简单,而且之前也提到过,就是当过渡是被如{{link-to}}transitionTo之类的提供了模型给路由的情况时,model钩子并不会被调用。

查看afterModelAPI文档

更多资源