我开始学习JavaScript承诺,但我无法理解如何将Q函数应用于Node.js回调函数。
建议{p> In similar question使用Q.denodeify()
,但它适用于fs.readFile()
,不适用于fs.exists()
。
这是一个简单的函数,它返回文件列表及其在目录中的大小:
function openDir(path, callback) {
path = __dirname + path;
fs.exists(path, function (exists) {
if (exists) {
fs.readdir(path, function (err, files) {
if (err) {
throw err;
}
var filesCounter = 0,
filesTotal = files.length,
result = [];
files.forEach(function (filename, index) {
fs.stat(path + filename, function (err, stats) {
if (err) {
throw err;
}
result[index] = {};
result[index].name = filename;
result[index].size = stats.size;
if (++filesCounter === filesTotal) {
callback(result);
}
});
});
});
}
});
}
http.createServer(function (req, res) {
openDir('/', function (data) {
res.writeHead(200, {
'Content-Type': 'application/json; charset=utf-8'
});
res.end(JSON.stringify(data));
});
}).listen(1337, '127.0.0.1');
如何使用promises(使用Q或其他库)编写此代码?
答案 0 :(得分:6)
fs.exists是一种无用的方法。它不返回节点样式回调,通常可以省略。在您的示例中,您将处理来自fs.readdir
的错误。
fs.exists()是一种时代错误,只是出于历史原因而存在。 几乎没有理由在你自己的代码中使用它。 node docs
以下是使用Q重新编写的示例,请注意我们通过返回promise而不是stats本身来将错误处理推迟到调用者。我提供了一个如何使用Q.defer处理fs.exists的示例,以展示如何代理非节点样式的回调。
var Q = require('q');
var fs = require('fs');
var http = require('http');
var readdir = Q.denodeify(fs.readdir);
function stat(file){
return Q.nfcall(fs.stat, file)
.then(function(stats){
stats.filename = file;
return stats;
});
}
function openDir(path) {
path = __dirname + path;
return readdir(path)
.then(function(files) {
return files.map(function(file){
return stat(path + file); }); })
.then(Q.all);
};
// proxy fs.exists to return a promise. Example of Q.defer
function exists(file){
var deferred = Q.defer();
fs.exists(file, function(result){
return result ? deferred.resolve(file) : deferred.reject('invalid file');
});
return deferred.promise;
}
http.createServer(function (req, res) {
openDir('/').done(
function (data) {
res.writeHead(200, {
'Content-Type': 'application/json; charset=utf-8'});
res.end(JSON.stringify(data));
},
function (err) {
res.writeHead(500, {
'Content-Type': 'application/json; charset=utf-8'});
res.end(JSON.stringify({error: err}));
});
}).listen(1337, '127.0.0.1');