节点js - 如何处理多个异步任务

时间:2016-09-26 14:32:44

标签: node.js asynchronous task

我是节点的新手并且无法处理多个异步任务。

除了节点之外,我还有另一台服务器(S1),它不会立即将数据返回给请求,它可以返回多种类型的数据,也可以发送通知而无需专门请求它们,因此节点必须监听数据从中解析并采取相应行动。

与此服务器(S1)的连接使用:

完成
S1 = net.createConnection({'host':S1Host, 'port': S1Port});

节点使用以下命令监听数据:

S1.on('data', function(data){
  S1DataParse(data);
});

我必须将正确的数据(在解析之后)路由到特定的POST请求。

app.post('/GetFooFromS1', function(req, res){

  // Send request to S1
  S1.write({'type':'foo'});

  // If got the correct data sometime in the future, send response to the browser
  res.setHeader('Content-Type', 'application/json');
  res.json({'status':'success', 'value':S1FooData});
});

我尝试使用异步模块,但没有成功。 我想做什么:

var asyncTasks = [];

app.post('/GetFooFromS1', function(req, res){

  asyncTasks.push(function(callback){
    // Send request to S1
    S1.write({'type':'foo'});
  });

  async.parallel(asyncTasks, function(response){
    res.setHeader('Content-Type', 'application/json');
    res.json({'status':'success', 'value':response});
  });
});

和S1DataParse中的另一个任务:

function S1DataParse(){
  if(data.type='foo'){
    asyncTasks.push(function(callback){
      callback(data);
    });
  }
}

但是,当然,第二个任务从未添加到asyncTasks数组中。我真的很困惑。 你可以帮帮我吗?

由于

- = - = - = - 编辑 - = - = - = -

最终,我遇到了事件 EventEmitter()

在POST请求中,我调用向数据服务器发送请求的函数( DataServerClientGet )。 在这个函数中,我注册了一个将获得未来数据的监听器。 eventEmitter.on('getData',returnDataServerData);

除了一件事,这一切都很有效。每当我刷新页面或添加其他POST请求时,我都会收到错误:

错误:发送后无法设置标头。

如果我解决这个问题会很棒。请帮帮我。

谢谢;)

整个代码如下所示:

var express = require('express');
var app = express();
var http = require('http').Server(app);
var bodyParser = require('body-parser')
var net = require('net');
var events = require('events');

var dataServerHost = '127.0.0.1';
var dataServerPort = 12345;
var dataServerClient;
var logMsg;

var eventEmitter = new events.EventEmitter();


/*******************************************/
//                Init
/*******************************************/
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({extended: true}));
app.use(express.static(__dirname + '/public'));


/*******************************************/
//       Connect to the data server
/*******************************************/
DataServerConnect();


/*******************************************/
// Open listener on port 3000 (to browser)
/*******************************************/
http.listen(3000, function(){
  logMsg = 'listening on *:3000';
  console.log(logMsg);
});


/*******************************************/
//                Routing
/*******************************************/
app.get('/', function(req, res){
  res.sendFile(__dirname + '/index.html');
});

app.post('/GetDataFoo', function(req, res){
  var msg;
  var size;

  msg ='\n{"Type":"Query", "SubType":"GetDataFoo","SearchFilter":""}';
  size = msg.length;

  logMsg = 'Client to DataServer: GetDataFoo';
  console.log(logMsg);

  DataServerClientGet('GetDataFoo', size, msg, res);
});


/*******************************************/
//               Functions
/*******************************************/
function DataServerConnect(){
  dataServerClient = net.createConnection({'host':dataServerHost, 'port': dataServerPort}, function(){
    logMsg = 'Connected to DataServer ['+dataServerHost+':'+dataServerPort+']';
    console.log(logMsg);
  });

  dataServerClient.on('data', function(data){

    logMsg = 'DataServerData>>>\n'+data.toString()+'DataServerData<<<';
    console.log(logMsg);

    DataServerDataParse(data.toString());
  });

  dataServerClient.on('end', function(){
    logMsg = 'Disconnected from DataServer';
    console.log(logMsg);
  });
}

function DataServerClientGet(type, size, msg, res){
  dataServerClient.write('Type: Json\nSize: '+size+'\n\n'+msg, function(err){

  var returnDataServerData = function returnDataServerData(results){
    res.setHeader('Content-Type', 'application/json');
    res.json({'status':'success', 'value':results});
  }
  eventEmitter.on('getData', returnDataServerData);
}

function DataServerDataParse(json){
  if(json.Type=='GetDataFoo')
  {
    var MessageList = json.MessageList;
    eventEmitter.emit('getData', MessageList);
  }
}

- = - = - = - 编辑 - = - = - = -

错误:发送后无法设置标头。每次调用 DataServerClientGet 时添加相同类型的同一个侦听器并且res是发送多次。

我通过添加以下内容解决了这个问题: removeListener(event,listener) 在res之后,在函数内部。无论如何,我认为这是错误的,如果有多个调用 DataServerClientGet 具有相同的类型等,可能会导致问题。

1 个答案:

答案 0 :(得分:0)

有一个可选的回调参数可以传递给写函数(docs),类似于:

S1.write({'type':'foo'},function(err){
  if(err){
   //Handle error
  }else{
     res.setHeader('Content-Type', 'application/json');
     res.json({'status':'success', 'value':response});
  }
})

这可以用于后期路由,但是在“数据”监听器中,当客户端没有初始化连接时,您无法将数据从服务器发送到客户端(它不是双向的)如果您想要双向行为,则可以检查{{ 3}}