Angularjs $ q.all

时间:2014-01-23 14:28:25

标签: angularjs promise q

我在angularjs中实现了$ q.all,但我无法使代码工作。这是我的代码:

UploadService.uploadQuestion = function(questions){

        var promises = [];

        for(var i = 0 ; i < questions.length ; i++){

            var deffered  = $q.defer();
            var question  = questions[i]; 

            $http({

                url   : 'upload/question',
                method: 'POST',
                data  : question
            }).
            success(function(data){
                deffered.resolve(data);
            }).
            error(function(error){
                deffered.reject();
            });

            promises.push(deffered.promise);
        }

        return $q.all(promises);
    }

这是我的控制器调用服务:

uploadService.uploadQuestion(questions).then(function(datas){

   //the datas can not be retrieved although the server has responded    
}, 
function(errors){ 
   //errors can not be retrieved also

})

我认为在我的服务中设置$ q.all存在一些问题。

3 个答案:

答案 0 :(得分:222)

在javascript中,只有 block-level scopes function-level scopes

阅读有关javaScript Scoping and Hoisting的文章。

了解我如何调试代码:

var deferred = $q.defer();
deferred.count = i;

console.log(deferred.count); // 0,1,2,3,4,5 --< all deferred objects

// some code

.success(function(data){
   console.log(deferred.count); // 5,5,5,5,5,5 --< only the last deferred object
   deferred.resolve(data);
})
  • 当你在for循环中写var deferred= $q.defer();时,悬挂到函数的顶部,这意味着javascript在for loop之外的函数范围内声明了这个变量
  • 对于每个循环,最后一个延迟覆盖前一个循环,没有块级范围来保存对该对象的引用。
  • 当调用异步回调(成功/错误)时,它们仅引用最后一个延迟对象并且仅解析它,因此 $ q.all永远不会被解析,因为它仍然等待其他延期对象。
  • 您需要为您迭代的每个项目创建一个匿名函数。
  • 由于函数确实有作用域,因此即使在执行函数后,对 closure scope 的引用也会保留。
  • 由于#dfsq评论:由于$ http本身返回一个promise,因此无需手动构造新的延迟对象。

angular.forEach的解决方案:

这是一个演示插件:http://plnkr.co/edit/NGMp4ycmaCqVOmgohN53?p=preview

UploadService.uploadQuestion = function(questions){

    var promises = [];

    angular.forEach(questions , function(question) {

        var promise = $http({
            url   : 'upload/question',
            method: 'POST',
            data  : question
        });

        promises.push(promise);

    });

    return $q.all(promises);
}

我最喜欢的方法是使用Array#map

这是一个演示插件:http://plnkr.co/edit/KYeTWUyxJR4mlU77svw9?p=preview

UploadService.uploadQuestion = function(questions){

    var promises = questions.map(function(question) {

        return $http({
            url   : 'upload/question',
            method: 'POST',
            data  : question
        });

    });

    return $q.all(promises);
}

答案 1 :(得分:35)

$ http也是一个承诺,你可以让它变得更简单:

return $q.all(tasks.map(function(d){
        return $http.post('upload/tasks',d).then(someProcessCallback, onErrorCallback);
    }));

答案 2 :(得分:12)

问题似乎是,当deffered.promise本身就是您应该添加的承诺时,您正在添加deffered

尝试更改为promises.push(deffered);,这样就不会将解包的承诺添加到数组中。

 UploadService.uploadQuestion = function(questions){

            var promises = [];

            for(var i = 0 ; i < questions.length ; i++){

                var deffered  = $q.defer();
                var question  = questions[i]; 

                $http({

                    url   : 'upload/question',
                    method: 'POST',
                    data  : question
                }).
                success(function(data){
                    deffered.resolve(data);
                }).
                error(function(error){
                    deffered.reject();
                });

                promises.push(deffered);
            }

            return $q.all(promises);
        }