我想创建一个程序,在其中读取目录内的文件,然后在其他目录中 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/