我刚刚遇到这种可怕的情况,我有一个字符串数组,每个字符串代表一个可能存在的文件(例如var files = [‘file1′,’file2′,’file3’].我需要遍历这些文件名并尝试看它是否存在于当前目录中,如果存在,则停止循环并忘记其余剩余文件.所以基本上我想找到那些的第一个现有文件,如果没有找到则回退到硬编码消息.
这就是我目前所拥有的:
var found = false; files.forEach(function(file) { if (found) return false; fs.readFileSync(path + file,function(err,data) { if (err) return; found = true; continueWithStuff(); }); }); if (found === false) { // Handle this scenario. }
这是不好的.它阻塞(readFileSync)因此它很慢.
我不能只为fs.readFile提供回调方法,它不是那么简单,因为我需要获取第一个找到的项目……并且可以以任意随机顺序调用回调.我认为一种方法是使用一个回调来增加一个计数器并保留一个找到/未找到信息的列表,当它到达files.length计数时,它会检查找到/未找到的信息并决定下一步做什么.
这很痛苦.我确实看到了事件IO的性能优势,但这是不可接受的.我有什么选择?
解决方法
不要在普通服务器环境中使用同步内容 – 事物是单线程的,这会在等待这个io绑定循环的结果时完全锁定. CLI实用程序=可能很好,服务器=启动时才可以.
异步流控制的通用库是
https://github.com/caolan/async
async.filter(['file1','file2','file3'],path.exists,function(results){ // results now equals an array of the existing files });
如果你想说,避免额外调用path.exists,那么你可以很容易地编写一个“first”函数来执行操作,直到某些测试成功.与https://github.com/caolan/async#until类似 – 但您对输出感兴趣.