您的位置:首页 > Web前端 > Node.js

利用promise写出更加优美的nodejs程序

2015-08-12 13:36 776 查看
http://my.oschina.net/sundq/blog/227410

摘要 nodejs所有的操作都是异步和callback模式,这样做的坏处是很容易造成或掉函数调用大坑,本文介绍如何使用promise写出更简洁明了的程序。
nodejs promise callback hell

目录[-]

什么是promise

将一般的异步调用的函数转换为promise风格的函数

使用方法


什么是
promise

promise
一个标准,它描述了异步调用的返回结果,包括正确返回结果和错误处理。关于详细的说明文档可以参考Promises/A+。目前实现
promise
标准的模块有很多,如Q]bluebird]Deferred,下面我们以Q为例,介绍一下
promise
nodejs
中的使用方法。


将一般的异步调用的函数转换为
promise
风格的函数

比如一般情况下,我们异步读取文件会写出下面的代码:
fs.readFile("test.txt",function(err,data){
if(!err){
console.log(data);
}else{
console.error(err);
}
});


我们可以把
fs.readFile
函数封装为
promise
风格的函数,如下:
var preadFile = function(file){
fs.readFile(file,function(err,data){
var deferred = Q.defer();
if(!err){
deferred.resolve(data);
}else{
deferred.reject(err);
}

return deferred.promise;
});
}


这样我们就可以这样调用这个方法了:
//then 的第一个参数是正确处理函数,第二个参数是错误处理函数
preadFile("test.txt").then(console.log,console.error);


可以看到这样调用更加直接明了。


使用方法

各个回调函数顺序传递数据:

nodejs
中我们回去数据时,时常遇回调函数的返回结果需要传递给下一个回调函数,到时很大的回调函数嵌套,
promise


可以很轻松的在回调函数之间传递参数,看下面的例子:
var fun1 = function (data,cb) {
cb(null,data+" fun1");
}

var fun2 = function (data,cb) {
cb(null,data+" fun2");
}

var fun3 = function (data,cb) {
cb(null,data+" fun3");
}

function main(data,cb){
fun1(data,function(err,data){
if(!err){
fun2(data,function(err,data){
if(!err){
fun3(data,cb);
}else{
cb(err);
}
});
}else{
cb(err);
}
});
}


可以看到
main
函数为了得到数据,需要嵌套调用
fun1
,
fun2
fun3
,如果需要调用很多函数,那么会形成很大的回调函数嵌套调用,导致代码看起来很丑陋,而且不容易维护,下面我们改用
promise
重写这段代码:
var Q = require("q");
var fun1 = function (data) {
var deferred = Q.defer();
deferred.resolve(data+" fun1");
return deferred.promise;
}

var fun2 = function (data) {
var deferred = Q.defer();
deferred.resolve(data+" fun2");
return deferred.promise;
}

var fun3 = function (data) {
var deferred = Q.defer();
deferred.resolve(data+" fun3");
return deferred.promise;
}

function main(data,cb){
fun2("test")
.then(fun3)
.then(fun4)
.done(function(data){
cb(null,data);//ok 获得的最终数据为 --->"test fun1 fun2 fun3"
},function(err){
cb(err);//failed
});
}


收集各个回调函数产生的数据:有时候我们需要执行很多回调函数,然手把这个回调函数的数据一齐传递给一个函数处理,此时我们可以使用
all
spread
方法,参看如下代码:
var Q = require("q");
var fun1 = function (data) {
var deferred = Q.defer();
deferred.resolve(data+" fun1");
return deferred.promise;
}

var fun2 = function (data) {
var deferred = Q.defer();
deferred.resolve(data+" fun2");
return deferred.promise;
}

var fun3 = function (data) {
var deferred = Q.defer();
deferred.resolve(data+" fun3");
return deferred.promise;
}

Q.all([
fun2("test1"),fun3("test2"),fun4("test3")
]).spread(function(){
console.log(arguments);//获得的参数为('test1 fun1', 'test2 fun2', 'test3 fun3' )
});


统一处理错误:传统的回调函数方式使用的话,我们需要在每一个回调函数里判断是否有错误需要处理,这样会存在很多冗余代码,使用
promise
的话,可以使用
done
或者
fail
统一在一个函数中处理错误,如第一个例子一样,使用
done
方法的第二个参数处理错误。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: