1

这里有一些您可能知道或不知道的 javascript 函数参数变量的有趣行为:

function foo(bar) {
    console.log('bar was:', bar); 
    arguments[0] = 'zap';        
    console.log('bar now:', bar); 
}  

foo('bam'); 
// bar was: bam
// bar now: zap

如您所见bar,变量现在指向另一个值。

我想以一种有点奇怪的方式利用这种行为,我想知道是否有可能以某种方式从函数范围之外更改参数值?

也许使用 call/apply 或其他一些棘手的 js 功能?

所以我可以在调用函数参数后更改它的值,例如:

function chooseNumber(number) { 
    setInterval(function() { 
        console.log('I choosed:', number) 
    }, 1000) 
}

chooseNumber(1);
// I choosed: 1
// I choosed: 1
// I choosed: 1
// ...

然后,如果我决定改变主意,我该怎么做才能使初始函数输出:

// I choosed: 2
// I choosed: 2
// I choosed: 2
// ...
4

2 回答 2

4

不,这应该是不可能的,尤其是在 EcmaScript 5 严格模式下。然而,hackery 可以在非严格模式下成功。以下尝试通过存储的arguments对象修改闭包中的参数;适用于火狐。

var argsave, bar;

(function foo(a) {
    argsave = arguments;
    bar = function () {
        alert("a is now: " + a);
    };
} (13));

bar(); // --> a is 13
argsave[0] = 42;
bar(); // --> a is 42

但是你不需要它,也不应该使用它;如果要修改闭包中的值,请在闭包中使用函数:

var setA, bar;

(function foo(a) {
    bar = function () {
        alert("a is now: " + a);
    };
    setA = function (newA) {
        a = newA;
    };
} (13));

bar(); // --> a is 13
setA(42);
bar(); // --> a is 42
于 2013-08-17T15:19:16.940 回答
0

更好的方法是清除先前的间隔,然后使用新行为再次注册您的 setInterval,如下所示。

function chooseNumber(number) { 
    var int = setInterval(function() { 
        console.log('I choosed:', number) 
    }, 1000) ; 
    return int;
}

var interval = chooseNumber(10);

if(someCondition ){
    clearInterval(interval);
    interval = chooseNumber(20);

}
于 2013-08-17T15:22:58.960 回答