node.js中实现同步操作的3种实现方法

众所周知,异步是得天独厚的特点和优势,但同时在程序中同步的需求(比如控制程序的执行顺序为:func1 -> func2 ->func3 )也是很常见的。本文就是对这个问题记录自己的一些想法。

需要执行的函数:

  1. var func1 = function(req,res,callback){
  2.   setTimeout(function(){
  3.     console.log('in func1');
  4.     callback(req,res,1);  
  5.   },13000);
  6. }
  7. var func2 = function(req,res,callback){ 
  8.   setTimeout(function(){
  9.     console.log('in func2');
  10.     callback(req,res,2);
  11.   },5000);
  12. }
  13.  
  14. var func3 = function(req,res,callback){
  15.   setTimeout(function(){
  16.     console.log('in func3');
  17.     callback(req,res,3);   
  18.   },1000);
  19. }

可以看出在func1,func2和func3中都是用了setTimeout函数,执行的时间分别为13秒,5秒和1秒。由于nodejs异步的特性,如果使用普通的函数调用方法:

  1. var req = null;
  2. var res = null;
  3. var callback = function(){};
  4. func1(req,res,callback);
  5. func2(req,res,callback);
  6. func3(req,res,callback);

输出内容:

  1. in func3
  2. in func2
  3. in func1

原因是因为nodejs是异步的,func2不会等func1执行完毕后再执行,而是立即执行(func3也是如此)。由于func3的运行时间最短而率先结束,func2次之,func1最后。但这明显不是我们想要的结果。怎么办?

解决办法一:callback

  1. //深层嵌套
  2. var req = null;
  3. var res = null;
  4.  
  5. func1(req,res,function(){
  6.   func2(req,res,function(){
  7.     func3(req,res,function(){
  8.       process.exit(0);   
  9.     })  
  10.   });  
  11. });

这种方法虽然能快速的解决,但暴露的问题也很明显,一是代码维护不方面,二是代码的深层嵌套看起来很不舒服。这种方法并不可取。

解决方法二:递归调用

  1. function executeFunc(funcs,count,sum,req,res){
  2.   if(count == sum){
  3.      return ; 
  4.    }
  5.    else{
  6.     funcs[count](req,req,function(){
  7.        count++;
  8.        executeFunc(funcs,count,sum,req,res);
  9.     });
  10.    }  
  11. }
  12.  
  13. //同步调用
  14. var req = null;
  15. var res = null;
  16. var funcs = [func1,func2,func3];
  17. var len = funcs.length;
  18. executeFunc(funcs,0,len,req,res);

先将多个函数组成一个数组。再可以利用递归函数的特性,使程序按照一定的顺序执行。

解决方法三:调用类库 

随着nodejs的发展,响应的类库也越来越多。Step和async 就是其中不错的。

1.Step的调用相对比较清爽:

  1. Step(
  2.   function thefunc1(){
  3.     func1(this);
  4.   },
  5.   function thefunc2(finishFlag){
  6.     console.log(finishFlag);
  7.     func2(this);
  8.   },
  9.   function thefunc3(finishFlag){
  10.     console.log(finishFlag);
  11.   }
  12. );

2.async 的 series方法,就本例而言,它的调用方法:

  1. var req = null;
  2. var res = null;
  3. var callback = function(){};
  4.  
  5. async.series(
  6.   [
  7.     function(callback){
  8.       func1(req,res,callback);
  9.     },  
  10.     function(callback){
  11.       func2(req,res,callback);
  12.     },
  13.     function(callback){
  14.       func3(req,res,callback);  
  15.     } 
  16.   ]
  17. );
原文地址:https://www.cnblogs.com/saryli/p/6912245.html