What does the "this" keyword refer to in the following Javascript code?

102 Views Asked by At

This is an example from JavaScript:The Definitive Guide by David Flanagan

//Replace the method named m of the object o with a version that logs messages before and after invoking the original method.

function trace(o, m) {
  var original = o[m];                            // Remember original method in the closure.
  o[m] = function() {                             // Now define the new method.
    console.log(new Date(), "Entering:", m);      // Log message.
    var result = original.apply(this, arguments); // Invoke original.
    console.log(new Date(), "Exiting:", m);       // Log message.
    return result;                                // Return result.
  };
}

I understand that since m is a method of o, 'this' should refer to the object o. But I cannot understand how; because in a function 'this' should refer to the global object (non-strict mode).
Also, how does the 'arguments' array contain the original function's arguments and not of the anonymous wrapper function's?

1

There are 1 best solutions below

1
On BEST ANSWER

The trace function will replace a method in an object with a wrapper, which will output some log meddages and call the original method.

The anonymous function is the wrapper that will be called instead of the original method. When it is called, it's called as a method of the object, so this will refer to the object.

Note that the anonymous function isn't executed in the trace function. It replaces the original method, so it will be called later on when the original method would have been called.

The arguments array is the arguments of the wrapper function, so that will be the arguments that was intended for the original method. That's why they are passed along when the original method is called.

If you have an object like this:

var html = {
  bold: function(x) {
    return "<strong>" + x + "</strong>";
  }
};

If you now use the trace function:

trace(html, 'bold');

Now the html object will contain the equivalent of:

{
  bold: function() {
    var m = 'bold';
    var original = function(x) {
      return "<strong>" + x + "</strong>";
    };
    console.log(new Date(), "Entering:", m);
    var result = original.apply(this, arguments);
    console.log(new Date(), "Exiting:", m);
    return result;
  }
}