node.js - 使用node.js+redis存储平均请求时间

标签 node.js redis

大家好,我有一个非常实用的 Redis 用例问题。假设我想使用以下 js 代码使用 Redis 存储平均请求时间。基本上我试图计算平均请求时间并在每个请求条目时保存到redis([req_path,req_time])

var rc=require('redis').createClient()
    ,rc2=require('redis').createClient()
    ,test_data=[
        ['path/1', 100]
        ,['path/2', 200]
        ,['path/1', 50]
        ,['path/1', 70]
        ,['path/3', 400]
        ,['path/2', 150]
    ];

rc.del('reqtime');
rc.del('reqcnt');
rc.del('avgreqtime');

for(var i=0, l=test_data.length; i<l; i++) {
    var item=test_data[i], req_path=item[0], req_time=item[1];
    console.log('debug: iteration # %d, item=%j', i, item);
    rc.zincrby('reqtime', req_time, req_path );
    rc.zincrby('reqcnt', 1, req_path, function(err, c) {
        rc2.zscore('reqtime', req_path, function(err, t) {
            var avg=t/c;
            console.log('req_path='+req_path+',t='+t+',c='+c);
            console.log('debug: added member %s to sorted set "avgreqtime" with score %f', req_path, avg);
            rc2.zadd('avgreqtime', avg, req_path);
        });
    });
}
rc.quit();
rc2.quit();

但是 avgreqtime 键没有按预期工作。从我得到的标准输出

debug: iteration # 0, item=["path/1",100]
debug: iteration # 1, item=["path/2",200]
debug: iteration # 2, item=["path/1",50]
debug: iteration # 3, item=["path/1",70]
debug: iteration # 4, item=["path/3",400]
debug: iteration # 5, item=["path/2",150]
req_path=path/2,t=undefined,c=1
debug: added member path/2 to sorted set "avgreqtime" with score %f NaN
req_path=path/2,t=undefined,c=1
debug: added member path/2 to sorted set "avgreqtime" with score %f NaN
req_path=path/2,t=undefined,c=2
debug: added member path/2 to sorted set "avgreqtime" with score %f NaN
req_path=path/2,t=undefined,c=3
debug: added member path/2 to sorted set "avgreqtime" with score %f NaN
req_path=path/2,t=undefined,c=1
debug: added member path/2 to sorted set "avgreqtime" with score %f NaN
req_path=path/2,t=undefined,c=2
debug: added member path/2 to sorted set "avgreqtime" with score %f NaN

redis 函数内的调试行在最后一次打印,而不是在每次迭代期间打印。我认为这与 Node.js 的异步特性有关,但我不知道如何完成这项工作。作为实验,我还尝试用以下内容替换 for 循环,但没有成功:

for(var i=0, l=test_data.length; i<l; i++) {
    var item=test_data[i], req_path=item[0], req_time=item[1];
    console.log('debug: iteration # %d, item=%j', i, item);
    rc.multi()
        .zincrby('reqtime', req_time, req_path )
        .zincrby('reqcnt', 1, req_path )
        .exec( function(err, replies) {
            console.log('debug(%s): got %j', req_path, replies);
            var avg=replies[0]/replies[1];
            rc2.zadd('avgreqtime', avg, req_path);
        });
}

这次我在每次迭代中得到了总请求时间,但问题是 req_path 坚持“path/2”,这是 test_data 中的最后一个 req_path。结果只有 'path/2' 被保存到 avgreqtime 并且这是错误的:

debug: iteration # 0, item=["path/1",100]
debug: iteration # 1, item=["path/2",200]
debug: iteration # 2, item=["path/1",50]
debug: iteration # 3, item=["path/1",70]
debug: iteration # 4, item=["path/3",400]
debug: iteration # 5, item=["path/2",150]
debug(path/2): got ["100","1"]
debug(path/2): got ["200","1"]
debug(path/2): got ["150","2"]
debug(path/2): got ["220","3"]
debug(path/2): got ["400","1"]
debug(path/2): got ["350","2"]

我使用的是Redis 2.4.5, Node redis客户端来自https://github.com/mranney/node_redis

最佳答案

您的猜测是正确的,这与 Node 的异步性质有关。我将在这里尝试一个简单的例子:

for(var i = 0; i < 10; i++) {
  someAsyncFunction(i, function(err, data) {
    console.log("executed function for", i);
  });
}

在这里,i将是您第一次引用它时所期望的内容(作为 someAsyncFunction 的参数)。 该函数的回调中,i永远是10 。当执行回调时,for 循环已经完成。要解决此问题,您需要绑定(bind) i不知何故。一种方法是使用匿名函数,立即执行:

for(var i = 0; i < 10; i++) {
  (function(i) {
    someAsyncFunction(i, function(err, data) {
      console.log("executed function for", i);
    });
  })(i); // Execute function with parameter i immediately
}

现在,i即使在回调内部也会绑定(bind)到正确的值。这不是最佳的,因为我们每次都需要指定一个新函数。这更好:

var executeTheFunction = function(i) {
  someAsyncFunction(i, function(err, data) {
    console.log("executed function for", i);
  });
};

for(var i = 0; i < 10; i++) {
  executeTheFunction(i);
}

请注意我们的executeTheFunction不接受回调。这意味着我们无法真正控制执行——所有调用都将立即执行,如果有很多调用,这可能不是我们想要的。在这种情况下,我推荐async module ,这使得这件事变得简单。

更新:以下是 async 的示例:

var calculateAverage = function(item, callback) {
    var req_path = item[0], req_time = item[1];

    rc.multi()
        .zincrby('reqtime', req_time, req_path )
        .zincrby('reqcnt', 1, req_path )
        .exec( function(err, replies) {
            if(err) return callback(err);
            console.log('debug(%s): got %j', req_path, replies);
            var avg=replies[0]/replies[1];
            rc2.zadd('avgreqtime', avg, req_path, callback);
        });
}

async.map(test_data, calculateAverage, function(err) {
    if(err)
        console.error("Error:", err);
    else
        console.log("Finished");
});

现在,您可以使用 async.queue 轻松管理此类内容等等

关于node.js - 使用node.js+redis存储平均请求时间,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9493105/

相关文章:

c - hiredis 命令对简单 C 代码的空回复

php - Laravel Homestead Redis 端口转发

mongodb - 哪个 nosql 选项相对于存储过程和大型数组?

go - 如何从接口(interface)中提取数据

c - 使用 valgrind 的 hiredis 中的内存泄漏

node.js - 不允许加载本地资源 - Chrome 上出现错误,Angular 应用程序在生产构建后未运行

javascript - 根据属性值从 ImmutableJS 列表中删除对象

javascript - Node.js 视频的完整路径

node.js - 已弃用的正文解析器?

javascript - Sails js,如何将后台事件发送到手机