Promises с fs и bluebird
В настоящее время я изучаю, как использовать promises в nodejs
поэтому моя первая задача состояла в том, чтобы перечислить файлы в каталоге, а затем получить содержимое каждого из этих двух шагов, используя асинхронные функции. Я придумал следующее решение, но имею сильное чувство, что это не самый элегантный способ сделать это, особенно в первой части, где я "превращаю" асинхронные методы в promises
// purpose is to get the contents of all files in a directory
// using the asynchronous methods fs.readdir() and fs.readFile()
// and chaining them via Promises using the bluebird promise library [1]
// [1] https://github.com/petkaantonov/bluebird
var Promise = require("bluebird");
var fs = require("fs");
var directory = "templates"
// turn fs.readdir() into a Promise
var getFiles = function(name) {
var promise = Promise.pending();
fs.readdir(directory, function(err, list) {
promise.fulfill(list)
})
return promise.promise;
}
// turn fs.readFile() into a Promise
var getContents = function(filename) {
var promise = Promise.pending();
fs.readFile(directory + "/" + filename, "utf8", function(err, content) {
promise.fulfill(content)
})
return promise.promise
}
Теперь цепочка promises:
getFiles() // returns Promise for directory listing
.then(function(list) {
console.log("We got " + list)
console.log("Now reading those files\n")
// took me a while until i figured this out:
var listOfPromises = list.map(getContents)
return Promise.all(listOfPromises)
})
.then(function(content) {
console.log("so this is what we got: ", content)
})
Как я писал выше, он возвращает желаемый результат, но я уверен, что есть более элегантный способ этого.
Ответы
Ответ 1
Код можно сделать короче, используя обобщенное обещание и .map
:
var Promise = require("bluebird");
var fs = Promise.promisifyAll(require("fs")); //This is most convenient way if it works for you
var directory = "templates";
var getFiles = function () {
return fs.readdirAsync(directory);
};
var getContent = function (filename) {
return fs.readFileAsync(directory + "/" + filename, "utf8");
};
getFiles().map(function (filename) {
return getContent(filename);
}).then(function (content) {
console.log("so this is what we got: ", content)
});
Фактически вы можете обрезать это еще больше, так как функции больше не тянут свой вес:
var Promise = require("bluebird");
var fs = Promise.promisifyAll(require("fs")); //This is most convenient way if it works for you
var directory = "templates";
fs.readdirAsync(directory).map(function (filename) {
return fs.readFileAsync(directory + "/" + filename, "utf8");
}).then(function (content) {
console.log("so this is what we got: ", content)
});
.map
должен быть вашим методом хлеба и масла при работе с коллекциями - он действительно мощный, поскольку он работает для чего-либо из обещания для массива promises, который сопоставляет с дальнейшим promises с любым сочетанием прямых значений между ними.