转自:http://www.cr173.com/html/13900_1.html
众所周知,异步是nodejs中得天独厚的特点和优势,但同时在程序中同步的需求(比如控制程序的执行顺序为:func1 -> func2 ->func3 )也是很常见的。本文就是对这个问题记录自己的一些想法。
需要执行的函数:
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函数,执行的时间分别为13秒,5秒和1秒。由于nodejs异步的特性,如果使用普通的函数调用方法: var req = null; var res = null; var callback = function(){}; func1(req,res,callback); func2(req,res,callback); func3(req,res,callback); //输出内容: in func3 in func2 in func1
原因是因为nodejs是异步的,func2不会等func1执行完毕后再执行,而是立即执行(func3也是如此)。由于func3的运行时间最短而率先结束,func2次之,func1最后。但这明显不是我们想要的结果。怎么办?
//解决办法一:callback //深层嵌套 var req = null; var res = null; func1(req,res,function(){ func2(req,res,function(){ func3(req,res,function(){ process.exit(0); }) }); }); //这种方法虽然能快速的解决,但暴露的问题也很明显,一是代码维护不方面,二是代码的深层嵌套看起来很不舒服。这种方法并不可取。 //解决方法二:递归调用 function executeFunc(funcs,count,sum,req,res){ if(count == sum){ return ; } else{ funcs[count](req,req,function(){ count++; executeFunc(funcs,count,sum,req,res); }); } } //同步调用 var req = null; var res = null; var funcs = [func1,func2,func3]; var len = funcs.length; executeFunc(funcs,0,len,req,res);
先将多个函数组成一个数组。再可以利用递归函数的特性,使程序按照一定的顺序执行。
解决方法三:调用类库,随着nodejs的发展,响应的类库也越来越多。Step和async 就是其中不错的。
//1. Step的调用相对比较清爽: Step( function thefunc1(){ func1(this); }, function thefunc2(finishFlag){ console.log(finishFlag); func2(this); }, function thefunc3(finishFlag){ console.log(finishFlag); } ); //2.async 的 series方法,就本例而言,它的调用方法: 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); } ] );
其中step安装方式npm install step
//使用示例1: var Step = require('step') , fs = require('fs'); /** * 順序印出Step1 - 3 */ Step( function step1() { console.log('Step1...'); throw 'error..'; //這個會掉到step2的arguments[0] return 123; //有return才會往下走 }, function step2() { console.log('Step2...'); console.log(arguments); //可以觀察接到的參數 return 223; }, function step3() { console.log('Step3...'); console.log(arguments); } ); //使用示例2: var Step = require('step') , fs = require('fs'); /** * 順序印出Step1 - 3 */ Step( function step1() { console.log('Step1...'); //如果function有callback,需將callback以this替代 fs.readFile('/etc/hosts', 'utf8', this); }, function step2() { console.log('Step2...'); //觀察此時argument[1]將會接到callback的值 //(PS:此部份callback的參數數量與位置將會與step2的arguments對應) console.log(arguments); //可以觀察接到的參數 return 223; }, function step3() { console.log('Step3...'); console.log(arguments); } );
参考: