javascript - 有没有办法只用一个回调调用递归地进行异步调用?

标签 javascript express asynchronous recursion cloudant

我想知道是否有人知道在 javascript 中递归执行异步调用的方法。我正在尝试一次将 500 个项目的批量数据上传到 cloudant 数据库。我不知道 json 的文档长度(大约在 2000-4000 之间)所以我创建了一个递归函数将它分成 500 个 block 并上传。

insertDBBulkData: function(trgtDB, dbDocData, callback) {

    // if length > 500, recursively upload 500 at a time
    if(dbDocData.length > 500) {
        console.log("File too big. Length " + dbDocData.length)
        module.exports.insertDBBulkData(trgtDB, dbDocData.slice(500, dbDocData.length), function(err, body) {
            if(err) {   
                console.log(err);
                callback(err, body)
            } else {
                // only callback on last one
                callback(err, body);
            }
        });

        dbDocData = dbDocData.slice(0, 500);
    }

    trgtDB.bulk({"docs": dbDocData}, function(err, body) {
        if(err) {
            callback(err);
        } else {
            console.log("Successfully uploaded " + dbDocData.length + " users. ")
            callback(null, "Success!")
        }
    });
},

问题是:因为我不知道哪个调用会最后结束,所以我不知道何时将响应发送回服务器(我只能做一次)。我尝试过使用 Promises,但据我所知,我不能将 Promises 与这种递归方法一起使用,因为我没有固定次数调用该方法。这可以通过延迟 promise 来实现吗?

感谢任何帮助。 谢谢!

解决方案:

感谢 Jonas W。我能够在 for 循环中使用多个 promise 来执行此操作。它不使用递归,但效果更好。这是我的解决方案:

insertDBBulkData: function(trgtDB, dbDocData, callback) {
    const promises = [];

    for(let start = 0; start < dbDocData.length; start += 500) {
        console.log("File too big. Index: " + start);

        var dbBulkDataPromise = new Promise(function(resolve, reject) {
            trgtDB.bulk({"docs": dbDocData.slice(start, start+500)}, function(err, body) {
                if(err) {
                    reject(err);
                } else {
                    resolve("Success")
                }
            });
        });

        promises.push(dbBulkDataPromise);
    }

    Promise.all(promises).then(function(values) {
        console.log(values);
        var completed = true;

        for (message in values) {
            if (message != "Success") {
                completed = false;
            }
        }

        if (completed) {
            callback(null, values)
        } else {
            console.log("Partial upload")
            callback("Partial upload only", null)
        }


    }).catch(function(err) {
        console.log("Error uploading data: " + err);
        callback(err, null)
    });
},

最佳答案

如果您将批量插入包装到一个 promise 中:

function insert(docs) {
   return new Promise((resolve, reject) => {
      trgtDB.bulk({ docs }, function(err, body) {
        if(err) {
          reject(err);
        } else {
           console.log("Successfully uploaded " + docs.length + " users. ")
           resolve("Success!")
       }
   });
  });
}

现在您可以遍历大量数据,将其分成 block ,开始为每个 block 插入,然后等待所有的 promise :

const promises = [], size = 500;

for(let start = 0; start < dbDocData.length; start += size)
  promises.push(insert( dbDocData.slice(start, start + size) ));

Promise.all(promises).then(() => {
  console.log("all done");
}).catch(/* handle errors */);

关于javascript - 有没有办法只用一个回调调用递归地进行异步调用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51503704/

相关文章:

javascript - multer上传不工作

node.js - 使用http查询字符串作为数据库对象node.js/express

node.js - 使用 express 和 mongodb 提供动态 URL

javascript - 分离的选择器事件 VS 多个 if 语句

Javascript,以编程方式触发firefox中的文件下载

node.js - Mean.JS 中的路由

powershell - 如果我在 powershell 脚本中使用 System.Windows.Forms,则使用 System.Net.WebClient 异步上传文件的 Register-ObjectEvent 将停止工作

javascript - 是否保证事件处理程序在调用 AJAX 回调之前完成?

java - 是否有干净的方法将上下文数据传递给@Asynchronous ejb 调用?

javascript - 无法使用选中的 ="Checked"属性找到单选按钮元素