javascript - NODEjs 在 for 循环内同步 exec 语句

标签 javascript node.js exec fs

我想创建一个程序,在其中读取目录内的文件,然后在其他目录中 grep 这些名称。

这是代码:

var exec = require('child_process').exec;
var internalCount = 0;
var i = -1;
for (; i < fileNamesInDir.length / 2;) {
  if (i === internalCount) {
    continue;
  }
  (function () {
    i = internalCount;
    var j = i;
    exec('grep -nre js/' + fileNamesInDir[j] +
      ' ./ --exclude-dir=node_modules --exclude=searchResults.txt',
      function (error, stdout, stderr) {
        outputString += "\n";
        outputString +=
          "***********************************************************";
        outputString += "\n";
        outputString += "RESULTS FOR  ::" + fileNamesInDir[j];
        outputString += "\n";
        outputString +=
          "***********************************************************";
        outputString += "\n";
        outputString += stdout;
        outputString += "\n";
        if (++internalCount === fileNamesInDir.length) {
          writeToFile();
        }
      });
  })();
}

问题在于执行语句不同步,并且回调函数异步返回数据。为了解决这个问题,我采用了另一个变量,如果迭代索引与该变量相同,则继续 for 循环。但是,使用此代码,它会进入无限循环,并且不会触发 exec 语句。

最佳答案

Javascript 中的 for 循环同步运行,这里无法告诉它等待循环中的任何内容。如果你尝试各种 hack,你最终会因为 Javascript 的单线程特性而遇到障碍或无限循环。所以......基本上你不能让 for 循环等待循环内的异步操作完成,然后再进入循环的下一次迭代。因此,您必须以不同的方式解决问题。


下面是一种常见的异步迭代方案,其中操作是协调并一个接一个地完成的:

var i = 0;
var outputString = "";
function next() {
    if (i < fileNamesInDir.length) {
        exec('grep -nre js/' + fileNamesInDir[i] +
          ' ./ --exclude-dir=node_modules --exclude=searchResults.txt',
          function (error, stdout, stderr) {
            outputString += "\n";
            outputString +=
              "***********************************************************";
            outputString += "\n";
            outputString += "RESULTS FOR  ::" + fileNamesInDir[i];
            outputString += "\n";
            outputString +=
              "***********************************************************";
            outputString += "\n";
            outputString += stdout;
            outputString += "\n";
            i++;
            next();
        });
    } else {
        writeToFile();
    }
}
next();

这将连续运行每个 grep 操作,以便结果按顺序连接。


但是,您不必连续运行此类代码。只要您按如下顺序收集结果,就可以并行运行所有这些操作:

var exec = require('child_process').exec;
var results = [];
var doneCntr = 0;
fileNamesInDir.forEach(function(item, index) {
        exec('grep -nre js/' + item +
          ' ./ --exclude-dir=node_modules --exclude=searchResults.txt',
          function (error, stdout, stderr) {
            var outputString = "";
            outputString += "\n";
            outputString +=
              "***********************************************************";
            outputString += "\n";
            outputString += "RESULTS FOR  ::" + item;
            outputString += "\n";
            outputString +=
              "***********************************************************";
            outputString += "\n";
            outputString += stdout;
            outputString += "\n";
            results[index] = outputString;
            ++doneCntr;
            if (doneCntr === fileNamesInDir.length) {
                writeToFile(results.join(""));
            }
        });
    });
}

您的代码还缺少需要添加的错误处理。


我最喜欢的方法是使用 Promise,让 Promise.all() 为我们完成收集结果的工作:

function execP(cmd) {
    return new Promise(function(resolve, reject) {
        exec(cmd, function(err, stdout, stderr) {
            if (err) {
                reject(err);
            } else {
                resolve({stdout, stderr});
            }
        });
    });
}

Promise.all(fileNamesInDir.map(function(item) {
    return execP('grep -nre js/' + item +
          ' ./ --exclude-dir=node_modules --exclude=searchResults.txt').then(function(data) {
        return "\n" +
           "***********************************************************\n" + 
           "RESULTS FOR  ::" + item + "\n" + 
           "***********************************************************\n" +
           data.stdout + "\n";

    });
})).then(function(results) {
    var output = results.join("");
    // process output here
}, function(err) {
    // process error here
});

关于javascript - NODEjs 在 for 循环内同步 exec 语句,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38300835/

相关文章:

javascript - 如何设置选定的 li 值以使用 jquery 输入

javascript - 脚本438 : Object doesn't support property or method 'filter' IE10

node.js - 在node.js riak-js .getClient中返回空对象

Go 程序在调用另一个程序后结束并出现错误代码

javascript - 任何以基于标准的跨平台 JS 和 DOM 为目标的 JavaScript 框架?

javascript - Ajax请求从php文件获取数据?

javascript - Discord.js 中使用 MySQL 的黑名单系统

node.js - "TypeError: Cannot read property ' 为多个请求推送 ' of null"

php - 使用 php 运行外部程序

bash - 如何在查找表达式中使用带有 sed 的文件名