这篇文章主要介绍nodejs的常用模块async,nodejs是异步io的,回头看看自己写的代码,回调函数几乎无处不在,这时候,管理好你的回调函数,理清你自己的思路就变得无比重要。(如果你还停留在闷头写代码,只是为了实现功能,那建议你放空心态,推倒重来)
先看一段代码
var objs = [{name:'A'}, {name:'B'}, {name:'C'}];
function doSomething(obj, cb)
{
console.log("我在做" + obj.name + "这件事!");
cb(null, obj);
}
doSomething(objs[0], function(err, data){
if(err)
{
console.log('处理错误!');
}
else
{
doSomething(objs[1], function(err, data){
if(err)
{
console.log('处理错误!');
}
else
{
doSomething(objs[2], function(err, data){
if(err)
{
console.log('处理错误!');
}
else
{
console.log('处理成功!');
}
});
}
});
}
});
做A这件事
如果A成功,再做B这件事
如果B成功,再做C这件事
安装async模块,命令行下
npm install --save async
var async = require('async');
async.waterfall([
function(cb)
{
doSomething(objs[0], cb);
},
function(data, cb)
{
doSomething(objs[1], cb);
},
function(data, cb)
{
doSomething(objs[2], cb);
}
], function (err, result) {
if(err)
{
console.log('处理错误!');
}
else
{
console.log('处理成功!');
}
});
var async = require('async');
async.waterfall([
//A这件事
function(cb)
{
doSomething(objs[0], function(err, dataA){
console.log(dataA);
cb(err, dataA); //如果发生err, 则瀑布就完了,后续流程都不会执行,B和C都不会执行
});
},
//B这件事,dataA就是上一步cb(err, dataA)中传进来的dataA
function(dataA, cb)
{
doSomething(objs[1], function(err, dataB){
console.log(dataB);
cb(err, dataA, dataB); //如果发生err, 则瀑布就完了,后续流程都不会执行,C不会执行
});
},
//C这件事
function(dataA, dataB, cb)
{
doSomething(objs[2], function(err, dataC){
console.log(dataC);
cb(err, dataA, dataB, dataC);
});
}
], function (err, dataA, dataB, dataC) { //瀑布的每一布,只要cb(err, data)的err发生,就会到这
if(err)
{
console.log('处理错误!');
}
else
{
console.log('处理成功!');
console.log(dataA);
console.log(dataB);
console.log(dataC);
}
});
我在做A这件事!
{ name: 'A' }
我在做B这件事!
{ name: 'B' }
我在做C这件事!
{ name: 'C' }
处理成功!
{ name: 'A' }
{ name: 'B' }
{ name: 'C' }
下一个要介绍的模型,whilst,一直执行主函数,直到条件不满足,或者发生异常
var async = require('async');
var i = 0;
async.whilst(
//条件
function() {
return i < 3; //true,则第二个函数会继续执行,否则,调出循环
},
function(whileCb) { //循环的主体
console.log(i);
i++;
whileCb();
},
function(err) { //here 如果条件不满足,或者发生异常
console.log("err is:" + err);
console.log("end,the i is:" + i);
}
);
i = 0;
async.whilst(
function() {
return i < 3; //true,则第二个函数会继续执行,否则,调出循环
},
function(whileCb) { //循环的主体
console.log(i);
i++;
if(i == 2)
{
whileCb("It's time to break.");
}
else
{
whileCb();
}
},
function(err) { //here 如果条件不满足,或者发生异常
console.log("err is:" + err);
console.log("end,the i is:" + i);
}
);
0
1
2
err is:undefined
end,the i is:3
0
1
err is:It's time to break.
end,the i is:2
第三个,each,这个就不多说了,处理记录集是每一个程序员都要遇到的问题。看例子
var async = require('async');
var objs = [{name:'A'}, {name:'B'}, {name:'C'}];
function doSomething(obj, cb)
{
console.log("我在做" + obj.name + "这件事!");
cb(null, obj);
}
async.each(objs, function(obj, callback) {
doSomething(obj, function(){
callback();
});
}, function(err){
console.log("err is:" + err);
});
async.each(objs, function(obj, callback) {
doSomething(obj, function(){
callback("It's a err.");
});
}, function(err){
console.log("err is:" + err);
});
我在做A这件事!
我在做B这件事!
我在做C这件事!
err is:undefined
我在做A这件事!
err is:It's a err.
我在做B这件事!
我在做C这件事!
第二个each,相信90%的nodejs的新手程序员都不明白,因为nodejs是异步,函数必然阻塞。(同一函数同时只有一个所谓的线程执行它),也就是不存在所谓的线程安全问题(这句定义其实本身又存在问题,因为异步会导致必然的上一层共享数据的安全问题,也就是说,nodejs不会有同层次的线程安全问题),得先理解,什么是栈。
第四个,eachSeries,这个和each差不多,简单些,就是遍历每一个对象是分步的,上一个对象的callback执行完之后,在执行下一个。
var async = require('async');
var objs = [{name:'A'}, {name:'B'}, {name:'C'}];
function doSomething(obj, cb)
{
console.log("我在做" + obj.name + "这件事!");
cb(null, obj);
}
async.eachSeries(objs, function(obj, callback) {
doSomething(obj, function(){
callback();
});
}, function(err){
console.log("err is:" + err);
});
//和each是有明显区别的,如果没有异步调用,和each无差别,如果有异步调用,则区别十分大
async.eachSeries(objs, function(obj, callback) {
doSomething(obj, function(){
callback("It's a err.");
});
}, function(err){
console.log("err is:" + err);
});
我在做A这件事!
我在做B这件事!
我在做C这件事!
err is:undefined
我在做A这件事!
err is:It's a err.
看到第二个eachSeries了吧,第二个A执行了之后,抛出了异常,B,C是必然不会被执行的。
async是我日常工作中用的最多的一个工具模块了。这种帮我们理清程序和思路的工具,用熟悉一个就好。相信大家会走得更远。