Promises.all() 在具有 'promised' 属性 的对象数组上

Promises.all() on an array of objects with a 'promised' property

编辑:在提出这个问题后的短短几个小时内,我学到了足够多的知识,意识到这个问题是一个关于严重滥用承诺的完美案例研究:

我正在尝试将工作流转换为使用 promises,因为它是一个可怕的回调巢穴,而且我进展顺利,直到我得到一个特定的函数。我正在尝试使用 readFile() 的 promisified 版本来获取一系列文件的内容,但我还需要保留每个文件的文件名,以便将其添加到最终将被解析的对象中下一个step/function中的文件内容。只有在我遍历文件名列表(并创建对 return 文件内容的承诺)时才能访问此文件名。现在我有这个,但行不通:

    // Accepts a sequence of filenames and returns a sequence of objects containing article contents and their filenames
    function getAllFiles(filenames) {
      var files = []; // This will be an array of objects containing Promises on which we want to call .all()
      filenames.each( function (filename) {
        files.push({content: fsPromise.readFileAsync(articlesPath + '/' + filename, 'utf8'), 
                   filename: filename});    
      });

这是试图表达我接下来想做的事情:

      Promise.all(files).then( function(files) {
        return Lazy(files);
      });
    } 

换句话说,我想从promises数组中得到的是像{content: "FILE CONTENTS", filename: "fileN.txt"}这样的对象数组。第一部分是 readFileAsync() 的结果,因此这是一个在我可以继续下一步之前需要解决的承诺。据我所知,filenames.each() 调用是我将文件内容与其来源文件的名称相关联的唯一机会。我无法控制文件的内容,所以我无法将文件名存储在文件内容中,这将是丑陋的数据冗余,而且无论如何都是一个坏主意。 Lazy() 调用只是将完成的数组转换为 lazy.js 序列,因为这是我在项目其余部分中用于集合处理的库。

我正在考虑的半解决方案

是否可以从对象中提取 promise 部分并将它们存储在单独的数组中,然后调用 .all()?换句话说,当所有的承诺都完成后,我回到原来的数组并通过索引检索匹配的文件名?如果我能够用 references 填充第二个数组到原始承诺(即我知道原始承诺何时解决),我觉得这会起作用。这行得通吗?

或者,

我可以在 .each() 调用中进行同步吗?例如。

filenames.each( function(filename) {
  files.push({content: fsPromise.readFileAsync(articlesPath + '/' + filename, 'utf8')
              .then( function(fulfilledContent) {return fulfilledContent}), 
              filename: filename
});

(我知道我还需要处理这些承诺被拒绝而不是解决的可能性——一旦我想出我想做的事情是否可行,我会想办法做到这一点)

有两个简单的解决方案:

  • 访问上次回调中的filenames数组。它与 Promise.all 将满足的内容数组的顺序相同:

    Promise.all(filenames.map(function(filename) {
        return fsPromise.readFileAsync(articlesPath + '/' + filename, 'utf8');
    })).then(function(contents) {
        return Lazy(contents).map(function(content, i) {
            return {content: content, filename: filenames[i]};
        });
    })
    
  • 在遍历数组时,立即为包含文件名和内容的对象做出承诺:

    Promise.all(filenames.map(function(filename) {
        return fsPromise.readFileAsync(articlesPath + '/' + filename, 'utf8').then(function(content) {
            return {content: content, filename: filename};
        });
    })).then(Lazy);