【问题标题】:Recursive Promises?递归承诺?
【发布时间】:2014-02-04 08:26:54
【问题描述】:

我想迭代位于 HTML 5 文件系统中的所有文件,并在迭代完成后启动一些事件。由于这是异步 + 承诺,我很难理解它应该如何工作。

我正在使用 angularJS 并创建了一个服务来封装 html 5 文件系统的特定功能。

这是递归函数:

function walkDirectory(path) {

    fileSystem.getFolderContents(path) //this is the services and it returns a promise containing all files in the current folder or directory
        .then(function(entries) {

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

                if(entries[i].isFile) {
                    console.log("is File: " + entries[i].name);
                    //do something with file here
                } 
                else if (entries[i].isDirectory) {
                    console.log("is Dir: " + entries[i].name);
                    walkDirectory(entries[i].fullPath);
                }
            }
        });
};

理想情况下,我想像这样调用该函数,并让它返回一个承诺,一旦所有文件都被遍历,该承诺就会被执行。

walkDirectory("/").then( function() {
  console.log(done);
});

任何提示/想法如何实现?

一个想法是有一个承诺数组,并为每个文件/目录的数组添加一个新的承诺。我的尝试:

function walkDirectory(path) {

    var defer= $q.defer();
    var promises = [defer.promise];

    fileSystem.getFolderContents(path)
        .then(function(entries) {

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

                if(entries[i].isFile) {
                    console.log("is File: " + entries[i].name);
                    //do something with file here
                    defer.resolve();
                    promises.push(defer.promise);
                } 
                else if (entries[i].isDirectory) {
                    console.log("is Dir: " + entries[i].name);
                    promises.push(walkDirectory(entries[i].fullPath));
                }
            }
        });

    return $q.all(promises);
};

walkDirectory("/").then(function() {
    console.log("done");
});

这似乎不起作用,因为控制台中永远不会显示完成。

【问题讨论】:

标签: javascript angularjs asynchronous promise q


【解决方案1】:

在填充之前返回数组。

相反,您需要在then() 回调中返回$q.all(promises),并返回外部promise:

return fileSystem.getFolderContents(path).then(function(entries) {
    return $q.all(entries.map(function(e) {
        if (e.isFile) {
            // Do something
            return null;  // Don't wait for anything
        } else {
            // Do something
            return walkDirectory(e.fullPath);
        }
    }));
});

【讨论】:

  • 哦,好吧,这意味着我可以摆脱 var defer= $q.defer() 和 var promises = [defer.promise],因为它们没有被使用。无论如何,您的解决方案似乎有效。谢谢。你能解释一下你的思路吗?
  • @BuildingJarl:你不明白什么? github.com/kriskowal/q#combination
  • 给定解决方案的 walkDirectory 的实现是什么?
  • @cyrf: 这 walkDirectory.
  • @SLaks:哦!谢谢。 +1,但是在successHandler中返回$q.all是什么意思?我期待看到$q.all(mapStuff).then(function(allData) { outerPromise.resolve(allData); })。实际上,我不明白从 successHandler 中“返回任何东西”。
猜你喜欢
  • 1970-01-01
  • 2018-05-29
  • 2017-11-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2014-03-23
  • 2017-09-21
相关资源
最近更新 更多