链接多个可选的异步Ajax请求

时间:2016-12-05 18:59:46

标签: javascript angularjs asynchronous promise q

我正在使用Angular 1.5.8。我的应用中的视图需要相同3个ajax请求的不同组合。一些视图需要来自所有三个的数据,其他视图需要来自两个甚至一个单个端点的数据。

我正在开发一个管理此数据检索的功能,要求应用只调用一次端点。我希望根据需要将ajax请求称为,但仅在需要时。目前我已经创建了一个有效的功能,但似乎可以使用改进。

$rootScope中包含以下功能。它使用fetchData()函数按要求循环访问get请求。检索数据时,它将存储在全局变量$ rootScope.appData中,然后再次调用fetchData()。检索所有数据后,将解析延迟的承诺,并将数据返回给控制器。

$rootScope.appData = {};

$rootScope.loadAppData = function(fetch) {
  var deferred = $q.defer();

  function getUser() {
    $http
      .get('https://example.com/api/getUser')
      .success(function(result){
        $rootScope.appData.currentUser = result;
        fetchData();
      });
  }

  function getPricing() {
    $http
      .get('https://example.com/api/getPricing')
      .success(function(result) {
        $rootScope.appData.pricing = result;
        fetchData();
      });
  }

  function getBilling() {
     $http
       .get('https://example.com/api/getBilling')
       .success(function(result) {
         $rootScope.appData.billing = result;
         fetchData();
       });
  }

  function fetchData() {
    if (fetch.user && !$rootScope.appData.currentUser) {
      getUser();
    } else if (fetch.pricing && !$rootScope.appData.pricing) {
      getPricing();
    } else if (fetch.billing && !$rootScope.appData.billing) {
      getBilling();
    } else {
      deferred.resolve($rootScope.appData);
    }
  }

  if ($rootScope.appData.currentUser && $rootScope.appData.pricing &&$rootScope.appData.billing) {
    deferred.resolve($rootScope.appData);
  } else {
    fetchData();
  }

  return deferred.promise;
};

对象fetch作为属性提交,此对象显示要调用的ajax请求。对$rootScope.loadAppData()的示例调用仅请求用户定价数据,如下所示:

$rootScope.loadAppData({user: true, pricing: true}).then(function(data){
   //execute view logic. 
});

我想知道:

  1. 这些功能的链接是否应该以不同的方式完成? fetchData()函数是否足够,或者这是执行此功能的奇怪方法?
  2. 有没有办法同时调用所有需要的Ajax请求,但在解析promise之前等待所有必需的调用完成?
  3. $rootScope中存储这样的数据是不寻常的吗?
  4. 我知道此功能目前没有正确处理错误。这是我在使用此代码段之前添加的功能,但与我的问题无关。

2 个答案:

答案 0 :(得分:2)

不使用.success方法,而是使用.then方法和返回数据到其成功处理程序:

function getUserPromise() {
    var promise = $http
      .get('https://example.com/api/getUser')
      .then( function successHandler(result) {
          //return data for chaining
          return result.data;
      });
    return promise;
}

使用服务而不是$ rootScope:

app.service("myService", function($q, $http) {

    this.loadAppData = function(fetchOptions) {

        //Create first promise
        var promise = $q.when({});

        //Chain from promise
        var p2 = promise.then(function(appData) {
            if (!fetchOptions.user) {
                return appData;
            } else {
                var derivedPromise = getUserPromise()
                  .then(function(user) {
                    appData.user = user;
                    //return data for chaining
                    return appData;
                });
                return derivedPromise;
            );
        });

        //chain from p2
        var p3 = p2.then(function(appData) {
            if (!fetchOptions.pricing) {
                return appData;
            } else {
                var derivedPromise = getPricingPromise()
                  .then(function(pricing) {
                    appData.pricing = pricing;
                    //return data for chaining
                    return appData;
                });
                return derivedPromise;
            );
        });
        //chain from p3
        var p4 = p3.then(function(appData) {
            if (!fetchOptions.billing) {
                return appData;
            } else {
                var derivedPromise = getBillingPromise()
                  .then(function(user) {
                    appData.billing = billing;
                    //return data for chaining
                    return appData;
                });
                return derivedPromise;
            );
        });

        //return final promise
        return p4;
    }
});

上面的示例为空对象创建了一个承诺。然后它链接三个操作。每个操作都会检查是否需要提取。如果需要,执行提取并将结果附加到appData对象;如果不需要提取,则appData对象将传递给链中的下一个操作。

用法:

myService.loadAppData({user: true, pricing: true})
  .then(function(appData){
    //execute view logic. 
}).catch(functon rejectHandler(errorResponse) {
    console.log(errorResponse);
    throw errorResponse;
});

如果任何获取操作失败,将跳过链中的后续操作,并调用最终的拒绝处理程序。

因为调用promise的.then方法会返回一个新的派生promise,所以很容易创建一个promise链。可以创建任何长度的链,并且由于可以使用另一个承诺(将进一步推迟其解析)来解决承诺,因此可以在链中的任何点暂停/推迟承诺的解析。这样就可以实现功能强大的API。 - AngularJS $q Service API Reference - Chaining Promises

答案 1 :(得分:0)

在原帖中找到了回答问题2 的好方法。使用$q.all()允许promises同时执行,一旦完成就解析,或者在其中一个失败后立即失败。由于@georgeawg,我已将此逻辑添加到服务中。这是我的重写,将此代码放入服务中,同时运行所有调用:

  services.factory('appData', function($http, $q) {
    var appData = {};
    var coreData = {};

    appData.loadAppData = function(fetch) {
      var deferred = $q.defer();
      var getUser = $q.defer();
      var getPricing = $q.defer();
      var getBilling = $q.defer();

      if (!fetch.user || coreData.currentUser) {
        getUser.resolve();
      } else {
        $http
          .get('https://example.com/api/getUser')
          .success(function(result){
            coreData.currentUser = result;
            getUser.resolve();
          }).error(function(reason) {
            getUser.reject(reason);
          });
      }

      if (!fetch.billing || coreData.billing) {
        getBilling.resolve();
      } else {
         $http
           .get('https://example.com/api/getBilling')
           .success(function(result) {
             coreData.billing = result;
             getBilling.resolve();
           }).error(function(reason) {
             getBilling.reject(reason);
           });
      }

      if (!fetch.pricing || coreData.pricing) {
        getPricing.resolve();
      } else {
         $http
           .get('https://example.com/api/getPricing')
           .success(function(result) {
             coreData.pricing = result;
             getPricing.resolve();
           }).error(function(reason) {
             getPricing.reject(reason);
           });
      }

      $q.all([getPricing.promise, getUser.promise, getBilling.promise]).then(function(result) {
        deferred.resolve(coreData);
      }, function(reason){
        deferred.reject(reason);
      });

      return deferred.promise;
    };

    return appData;
  });