I have some code like this:
for(var id=0; id < message.receiver.length; id++){
var tmp_id = id;
zlib.gzip(JSON.stringify(message.json), function(err, buffer){
...
pushStatusPool[message.receiver[tmp_id]] = null; // fix memory leak
delete pushStatusPool[message.receiver[tmp_id]];
...
});
}
And I got a warning that using tmp_id
in closure may cause problem because it is a mutable variable.
How could I avoid that? I mean how could I send an immutable variable to callback since this is a for loop and I can not change code of zlib.gzip
? Or in other words, how could I pass a argument to a closure?
Creating closures in a loop with
var
(tmp_id
) being in the upper scope of the callback function is a common mistake that should be avoided due to thevar
not being block-scoped. Because of this, and because each closure, created in the loop, shares the same lexical environment, the variable will always be the last iterated value (i.e.message.receiver.length - 1
astmp_id
) when the callback function gets invoked. Your IDE detects this behavior and complains rightly.To avoid the warning, there are several solutions:
Replace
var
withlet
ensuring each created closure to have its own scopedtmp_id
defined in each iteration:Create a lexical environment in each iteration by leveraging IIFE like gennadi.w did.
Create a callback function in each iteration by using a factory function (
createCallback
):bind
the variable(s) on the callback function in which they get prepended to its parameters:If possible,
var
should be avoided as of ECMAScript 2015 due to such error-prone behaviors.here a simplification of user24359's great answer. This is the solution:
The above code logs a b and is the solution. The following code logs b b :
@user24359 answer is a good solution but you can simply replace the
var
keyword by thelet
keyword.becomes
See details here.
Edit : As Heriberto Juárez suggested it, it will only works for browsers that supports EcmaScript6.
I have faced the same problem and solved it slightly modifying the answer of user24359, by passing the id to the closure:
You need to create a scope to correctly capture
tmp_id
using a self-executing function. That's because the entire for loop is one scope, meaning each time through, you're capturing the same variable. So the callback will end up with the wrong ids, becausetemp_id
's value will get changed before the callback is called.I'd ignore (or shut off) the warning, though, which seems to be complaining that because
temp_id
is mutable, you might reassign it. That's sort of silly. If you really want to fix it, try using theconst
keyword instead ofvar
.I've faced the same problem in protractor. Solved it using following code -