`

一些nodejs实现同步操作想法实现

 
阅读更多

转自: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);
  }
);

 参考:

http://opennodes.arecord.us/md/step.md

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics