在示例中使用带有Node.js函数的promise

时间:2014-02-16 14:28:28

标签: javascript node.js promise q

我开始学习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或其他库)编写此代码?

1 个答案:

答案 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');