眾所周知,異步是nodejs中得天獨(dú)厚的特點(diǎn)和優(yōu)勢(shì),但同時(shí)在程序中同步的需求(比如控制程序的執(zhí)行順序?yàn)椋篺unc1 -> func2 ->func3 )也是很常見(jiàn)的。本文就是對(duì)這個(gè)問(wèn)題記錄自己的一些想法。
需要執(zhí)行的函數(shù):
var func1 = function(req,res,callback){ setTimeout(function(){ console.log('in func1'); callback(req,res,1); },13000); } var func2 = function(req,res,callback){ setTimeout(function(){ console.log('in func2'); callback(req,res,2); },5000); } var func3 = function(req,res,callback){ setTimeout(function(){ console.log('in func3'); callback(req,res,3); },1000); }
可以看出在func1,func2和func3中都是用了setTimeout函數(shù),執(zhí)行的時(shí)間分別為13秒,5秒和1秒。由于nodejs異步的特性,如果使用普通的函數(shù)調(diào)用方法:
var req = null; var res = null; var callback = function(){}; func1(req,res,callback); func2(req,res,callback); func3(req,res,callback);
輸出內(nèi)容:
in func3 in func2 in func1
原因是因?yàn)閚odejs是異步的,func2不會(huì)等f(wàn)unc1執(zhí)行完畢后再執(zhí)行,而是立即執(zhí)行(func3也是如此)。由于func3的運(yùn)行時(shí)間最短而率先結(jié)束,func2次之,func1最后。但這明顯不是我們想要的結(jié)果。怎么辦?
解決辦法一:callback
//深層嵌套 var req = null; var res = null; func1(req,res,function(){ func2(req,res,function(){ func3(req,res,function(){ process.exit(0); }) }); });
這種方法雖然能快速的解決,但暴露的問(wèn)題也很明顯,一是代碼維護(hù)不方面,二是代碼的深層嵌套看起來(lái)很不舒服。這種方法并不可取。
解決方法二:遞歸調(diào)用
function executeFunc(funcs,count,sum,req,res){ if(count == sum){ return ; } else{ funcs[count](req,req,function(){ count++; executeFunc(funcs,count,sum,req,res); }); } } //同步調(diào)用 var req = null; var res = null; var funcs = [func1,func2,func3]; var len = funcs.length; executeFunc(funcs,0,len,req,res);
先將多個(gè)函數(shù)組成一個(gè)數(shù)組。再可以利用遞歸函數(shù)的特性,使程序按照一定的順序執(zhí)行。
解決方法三:調(diào)用類庫(kù)
隨著nodejs的發(fā)展,響應(yīng)的類庫(kù)也越來(lái)越多。Step和async 就是其中不錯(cuò)的。
1. Step的調(diào)用相對(duì)比較清爽:
Step( function thefunc1(){ func1(this); }, function thefunc2(finishFlag){ console.log(finishFlag); func2(this); }, function thefunc3(finishFlag){ console.log(finishFlag); } );
2.async 的 series方法,就本例而言,它的調(diào)用方法:
var req = null; var res = null; var callback = function(){}; async.series( [ function(callback){ func1(req,res,callback); }, function(callback){ func2(req,res,callback); }, function(callback){ func3(req,res,callback); } ] );