Javascript:如何在函数内修改全局变量?

时间:2012-12-26 06:36:36

标签: javascript scope closures

以下是我正在处理的代码。我想知道为什么,当我在函数中更改变量ttt时,更改不会停留在函数之外?我已经在最顶层宣布它为var ttt = new Array;

另外,为什么我不能在函数中使用变量i

代码:

  client.on('connection', function()
    {
        var sss;
        var aaa;

            console.log('Connected');

        for (i = 0 ; i < 120 ; i++)
            ttt[i] = 0;

        for (i = 0 ; i < 9 ; i++)
        {
                client.getMatchHistory(434582, function(err, result)        
            {
                sss = JSON.stringify(result);
                var myObject = eval('(' + sss + ')');
                console.log (myObject.object.data[i].object.value);

                ttt[myObject.object.data[i].object.value]++;
            });

        }

            for (i = 0 ; i < 120 ; i++)
                console.log ("Record" + i + " Successes: " + ttt[i]);

    });

3 个答案:

答案 0 :(得分:2)

正如您所指出的,您的代码存在两个独立的问题,并且它们都有些相关。首先,全局修改ttt 。问题是你在修改之前检查它们。我怀疑client.getMatchHistory()正在进行异步调用。在您执行第三个for循环时,无法保证第二个for循环中的所有异步操作都将在执行时执行。

第二个问题是范围问题,但问题不在于全球范围。由于client.getMatchHistory()是异步的,因此一旦循环执行完毕,就会调用回调。完成循环后执行i的值将为10。可能这不是你想要的。您需要创建一个回调生成函数,该函数获取i的值,并返回一个可用作回调的函数。像这样:

function make_callback(i) {
  return function(err, result) {
    // The body of your callback in here
  };
}

然后你应该在循环体中使用它:

client.getMatchHistory(434582, make_callback(i))

这将捕获当前迭代中i的值,生成的回调将在执行时使用该值。这应该可以解决i的问题。

答案 1 :(得分:0)

首先,所有全局变量都是有效的“窗口”对象字段,因此您可以使用window.ttt来确保使用全局变量而不是本地变量。此代码工作,您是否在开发人员工具中尝试过?调试器对这种变量的存在有何看法?

至于变量i:当然,你可以使用它,但最好在本地使用它,定义'var i;'在函数的顶部不破坏全局命名空间。

答案 2 :(得分:0)

client.getMatchHistory可能是异步请求,你希望在循环之后,你将填充ttt数组,为了实现这个,你必须创建一个在最后一个循环步骤之后运行的处理程序:

var afterloop=function() {
    for (var i = 0 ; i < 120 ; i++)
       console.log ("Record" + i + " Successes: " + ttt[i]);
}
for (var i = 0 ; i < 120 ; i++)
        ttt[i] = 0;

var i_final=0;
for (var i = 0 ; i < 9 ; i++)
        {   
            var i=i;   //localise i
            client.getMatchHistory(434582, function(err, result)        
            {
                i_final++;
                sss = JSON.stringify(result);
                var myObject = eval('(' + sss + ')');
                console.log (myObject.object.data[i].object.value);

                ttt[myObject.object.data[i].object.value]++;
                if (i_final>8) {afterloop();}
            });

        }
示例中的

i_final计算已完成的请求,由于异步,它们可以按随机顺序完成,因此在决定运行i时无法引用afterloop(),当i_final计数超过上次执行的请求时,运行应在上次请求完成后执行的函数。

注意:请尽可能少地使用全球变量,在您使用全局i的代码中无任何理由