最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Passing variables to callbacks in Node.js - Stack Overflow

programmeradmin0浏览0评论

I'm quite new to node and am trying to create something that gets some server info. But here's my problem. I setup a config object (this will, in time, bee updated dynamically by events that occur) and then later in, in a function, I try and access a value in this object. (See code below)

So First, I setup my vars:

var util            = require('util'),
    child           = require('child_process'),
    config          = {};

which works okay. Then I load my config:

function loadConfig( )
{
    // Add some code for auto-loading of args
    config = {
        "daemons": [
            ["Apache", "apache2"],
            ["MySQL",  "mysqld"],
            ["SSH", "sshd"]
        ]
    };
}

and init that calling the function

loadConfig();

After that, I run my check on daemons.

function getDaemonStatus( )
{
    for(var i=0; i<config.daemons.length; i++)
    {

        child.exec( 'ps ax -o \'%c %P\' | awk \'{if (($2 == 1) && ($1 == "\'' +
            config.daemons[i][1] + '\'")) print $0}\'',
            function( error, stdout, stderr )
        {

            console.log(config.daemons[i]);
        });
    }
}

The response I get is:

undefined
undefined
undefined

I don't really want to use a GLOBAL variable, so can you guys think of another way to solve my problem?

Thanks! =]

I'm quite new to node and am trying to create something that gets some server info. But here's my problem. I setup a config object (this will, in time, bee updated dynamically by events that occur) and then later in, in a function, I try and access a value in this object. (See code below)

So First, I setup my vars:

var util            = require('util'),
    child           = require('child_process'),
    config          = {};

which works okay. Then I load my config:

function loadConfig( )
{
    // Add some code for auto-loading of args
    config = {
        "daemons": [
            ["Apache", "apache2"],
            ["MySQL",  "mysqld"],
            ["SSH", "sshd"]
        ]
    };
}

and init that calling the function

loadConfig();

After that, I run my check on daemons.

function getDaemonStatus( )
{
    for(var i=0; i<config.daemons.length; i++)
    {

        child.exec( 'ps ax -o \'%c %P\' | awk \'{if (($2 == 1) && ($1 == "\'' +
            config.daemons[i][1] + '\'")) print $0}\'',
            function( error, stdout, stderr )
        {

            console.log(config.daemons[i]);
        });
    }
}

The response I get is:

undefined
undefined
undefined

I don't really want to use a GLOBAL variable, so can you guys think of another way to solve my problem?

Thanks! =]

Share Improve this question asked Nov 6, 2011 at 12:00 Daniel Noel-DaviesDaniel Noel-Davies 5571 gold badge6 silver badges10 bronze badges 1
  • possible duplicate of Access outside variable in loop from Javascript closure – Felix Kling Commented Nov 6, 2011 at 12:19
Add a ment  | 

1 Answer 1

Reset to default 13

This is a gotcha that lots of people run into because of the asynchronous ordering of execution.

Your for loop will look from 0-3, and then exit when 'i' is four, obviously. The tough part to remember here is that your callback for exec won't run immediately. In only runs once the process has started, and by the time that happens, the for loop will be done.

That means that essentially, all three times that your callback function is running, you are essentially doing this:

console.log(config.daemons[4]);

That's why it prints 'undefined'.

You need to capture the 'i' value in a new scope, by wrapping the loop contents in an anonymous, self-executing function.

function getDaemonStatus( ) {
    for(var i=0; i<config.daemons.length; i++) {
        (function(i) {

             child.exec( 'ps ax -o \'%c %P\' | awk \'{if (($2 == 1) && ($1 == "\'' +
                config.daemons[i][1] + '\'")) print $0}\'',
                function( error, stdout, stderr ) {

                console.log(config.daemons[i]);
            });

        })(i);
    }
}

Also, I see that your function is called 'getDaemonStatus'. Just remember that, since that exec callback is asyncronous, that also means that you can't collect the results of each callback, and then return them from the getDaemonStatus. Instead, you will need to pass a your own callback, and call the it from inside your exec callback.

Updated

Note though, the easiest way to have a scope per-iteration is to use forEach, e.g.

function getDaemonStatus( ) {
    config.daemons.forEach(function(daemon, i){
         child.exec( 'ps ax -o \'%c %P\' | awk \'{if (($2 == 1) && ($1 == "\'' +
            daemon[1] + '\'")) print $0}\'',
            function( error, stdout, stderr ) {

            console.log(daemon);
        });
    }
}
发布评论

评论列表(0)

  1. 暂无评论