admin管理员组

文章数量:1344410

In javascript, you can rewrite a function, like so:

function foo() { 
   setTimeout(function() {
        alert('sup stallion');
        foo = function() { //rewrite foo to nolonger wait 4 seconds to alert.
             alert('sup stallion');
        }
   }, 4000);
}

Obviously this is a contrived example, but is there anything conceptually wrong with this approach (other than a race condition).

In javascript, you can rewrite a function, like so:

function foo() { 
   setTimeout(function() {
        alert('sup stallion');
        foo = function() { //rewrite foo to nolonger wait 4 seconds to alert.
             alert('sup stallion');
        }
   }, 4000);
}

Obviously this is a contrived example, but is there anything conceptually wrong with this approach (other than a race condition).

Share Improve this question edited Mar 27, 2012 at 19:38 Alan asked Mar 27, 2012 at 18:17 AlanAlan 46.9k20 gold badges117 silver badges138 bronze badges 1
  • 2 Why would there be a race condition? There aren't multiple simultaneous threads running the JavaScript. – mellamokb Commented Mar 27, 2012 at 18:25
Add a ment  | 

3 Answers 3

Reset to default 7

Self modifying code can be confusing, and hard to debug, so it's generally avoided.

Other than that there is no problem, and no race condition either.

One thing I noticed while testing your code. Consider this:

setInterval(foo, 6000);

Function foo is being passed to setInterval before it was modified, and the original foo will run every 6 seconds, even after the binding has been updated.

On the other hand, the code below will run the original function only on the first call (which updates the binding). Subsequent calls will invoke the updated version:

setInterval(function(){foo();}, 6000);

Looks obvious, but could be hard to debug...

Dynamic function rewriting can be used as a form of lazy initialization, however there is a catch:

function Foo() {...}
Foo.prototype = {
    bar: function () {
        //some initialized variables to close over
        var a, b, c, ...;
        Foo.prototype.bar = function () {
            //do stuff with variables
        };
        Foo.prototype.bar.call(this);
    }
};

While this code is relatively straight-forward to understand, and would be used as:

var f = new Foo();
f.bar(); //initializes `bar` function
f.bar(); //uses initialized `bar` function

it has a hidden issue:

var f = new Foo(),
    g = {};
//passing by reference before the function was initialized will behave poorly
g.bar = f.bar;
f.bar(); //initializes `bar` function
g.bar(); //re-initializes `bar` function
f.bar(); //uses initialized `bar` function
g.bar(); //re-initializes `bar` function

It's for this reason that any initialization needed for a function is typically done using a module pattern:

function Foo() {...}
Foo.prototype = {
    bar: (function () {
        var a, b, c, ..., fn;
        //some initialized variables to close over
        fn = function () {
            //do stuff with variables
        };
        return fn;
    }())
};

The module pattern has the disadvantage of calling the initialization code immediately, but wont have the issues associated with the function reference.

本文标签: Consequences of Dynamically rewriting a function in javascriptStack Overflow