3

我有一个在另一个函数内的 forEach 循环内的 ajax 调用。问题是,外部函数的回调在内部循环结束之前触发 - 因此“staticItemList”在传递给回调时没有填充项目。我该如何解决?我真的在这上面花了很多时间。谢谢。

  exports.buildTheList = function (parsedList, callback) {
        var staticItemList = {};
        parsedList.forEach(function(parsedItem) { 

                db.staticList.find({"_id":parsedItem.ID}).forEach(function(err, doc) {
                    if (!doc){
                        console.log("newStaticItem not found in db");
                        parsedDataFetcher.fetchParsedDetails(Path, function(parsedDetails){
                            staticItemList["_id"] = parsedItem.ID;
                            staticItemList[parsedItem.ID] = {
                                    "_id": parsedItem.ID,
                                    "type": parsedItem.type,
                                     } 
                                 })

                             }else {
                                   console.log("newStaticItem already found in db");
                             } 
                 });
             });


         callback(staticItemList);
     }
4

3 回答 3

5

在循环内添加一个计数器变量,并在每次异步方法完成时递减它。一旦计数器达到零,调用回调。在伪代码中:

 var counter = 0;
 foreach(function() {
     counter++;
     doAsync(function() {
           // add to list

           counter--;
           if (counter === 0) {
               callback(list);
           }
     });
 }

希望有帮助。

于 2012-11-26T20:31:04.510 回答
1

我会看类似 dojo promise 的东西all。从手册:

dojo/promise/all是一个函数,它接受多个 Promise 并返回一个新的 Promise,当所有 Promise 都已实现时,该 Promise 已实现。

该页面上的示例演示了如何执行所需的操作。

于 2012-11-26T20:36:54.093 回答
0

AJAX 异步发生,因此您的循环运行,触发所有 AJAX 调用,然后完成,调用您的回调。您可能需要在 foreach 循环中实现某种计数器,并在每个 AJAX 调用完成时检查它。一个完成的 AJAX 调用的数量与计数器的数量相同,您将从异步函数上下文中执行回调函数。

于 2012-11-26T20:34:32.543 回答