In this article we'll see what this means, and places where this is applicable.
We'll also look at how to manipulate this with built-in functions like
apply, call and bind.
Every JavaScript statement is executed in one execution context or the other. In it's simplest terms this refers to the "owner" of the function we currently executing. this helps us to get the object or the execution context we are currently working with.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
function func1() { | |
return this; | |
} | |
console.log(func1()); //returns Window object | |
function func2(){ | |
"use strict"; | |
return this; | |
} | |
console.log(func2()); //undefined |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var val = "window.val" | |
var obj = { | |
val: "obj.val", | |
innerMethod: function () { | |
var val = "obj.val.inner", | |
func = function () { | |
var self = this; | |
return self.val; | |
}; | |
return func; | |
}, | |
outerMethod: function () { | |
return this.val; | |
} | |
}; | |
//This actually gets executed inside window object | |
console.log(obj.innerMethod()()); //returns window.val | |
//Breakdown in to 2 lines explains this in detail | |
var _inn = obj.innerMethod(); | |
console.log(_inn()); //returns window.val | |
console.log(obj.outerMethod()); //returns obj.val |
Calling a method of a object
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var status = 1; | |
var helper = { | |
status : 2, | |
getStatus: function () { | |
return this.status; | |
} | |
}; | |
var theStatus1 = helper.getStatus(); //line1 | |
console.log(theStatus1); //2 | |
var theStatus2 = helper.getStatus; | |
console.log(theStatus2()); //1 |
Constructor
When defining a function to be used as a constructor with the new keyword, this can be used to refer to the object being created.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
function Person(name){ | |
this.personName = name; | |
this.sayHello = function(){ | |
return "Hello " + this.personName; | |
} | |
} | |
var person1 = new Person('Scott'); | |
console.log(person1.sayHello()); //Hello Scott | |
var person2 = new Person('Hugh'); | |
var sayHelloP2 = person2.sayHello; | |
console.log(sayHelloP2()); //Hello undefined | |
Function call
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
function testFunc() { | |
this.name = "Name"; | |
this.myCustomAttribute = "Custom Attribute"; | |
return this; | |
} | |
var whatIsThis = testFunc(); | |
console.log(whatIsThis); //window | |
var whatIsThis2 = new testFunc(); | |
console.log(whatIsThis2); //testFunc() / object | |
console.log(window.myCustomAttribute); //Custom Attribute |
If we don't provide a context by new (as explained above), whatIsThis defaults to reference the most global context it can find (window) and it will pollute the global object too. But whatIsThis2 acts as we expect since we are providing new there. Therefore make sure to put new keyword otherwise you'll be adding unnecessary variables to global context.
Event Handler
If the event handler is inline, this refers to global window object.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script type="application/javascript"> | |
function click_handler() { | |
alert(this); // alerts the window object | |
} | |
</script> | |
<button id='thebutton' onclick='click_handler()'>Click me!</button> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script type="text/javascript"> | |
function click_handler() { | |
alert(this); // alerts the button DOM node | |
} | |
function addhandler() { | |
document.getElementById('thebutton').onclick = click_handler; | |
} | |
window.onload = addhandler; | |
</script> | |
<button id='thebutton'>Click me!</button> |
Let's look at more complicated example
Here when you click thebutton it alerts undefined.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<script type="text/javascript"> | |
function Person(name) { | |
this.personName = name; | |
this.sayHello = function () { | |
alert("Hello " + this.personName); | |
} | |
} | |
function addhandler() { | |
var p1 = new Person('John'), | |
the_button = document.getElementById('thebutton'); | |
the_button.onclick = p1.sayHello; | |
} | |
window.onload = addhandler; | |
</script> | |
<button id='thebutton'>Click me!</button> | |
Problem: We've passed a reference to the sayHello method, which when executed as an event handler, runs in a different context than when it's executed as an object method. Here this refers to the DOM element. setTimeout exhibits the same behavior, delaying the execution of a function while at the same time moving it out into a global context.
Manipulating context with .apply and .call
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var obj1 = { | |
num : 4 | |
} | |
function multiply(val){ | |
return this.num * val; | |
} | |
console.log(multiply(3)); //impossible because no num defined in window (NaN) | |
console.log(multiply.apply(obj1, [4])); //16 | |
console.log(multiply.call(obj1, 4)); //16 |
The call method (also the apply method) allows us to specify the this object in which the function needs to get executed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<button id='thebutton'>Click me!</button> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
function Person(name) { | |
this.name = name; | |
this.sayHello = function () { | |
alert('Hello ' + this.name); | |
} | |
} | |
function addhandler() { | |
var p1 = new Person('Jason'), | |
the_button = document.getElementById('thebutton'); | |
the_button.onclick = p1.sayHello.call(p1); | |
} | |
window.onload = addhandler; |
Even though we apply .call there's a problem. call executes this immediately. Instead of providing a function reference to click handler, we are giving the result of an executed function.
Let's get introduced to .bind(). bind performs the same general task as call. altering the context in which a function executes. The difference is bind returns a function reference which can be used later rather than the result from immediate execution which call does. Let's see a basic example.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
this.statusId = 1; | |
var module = { | |
statusId : 4, | |
getStatus : function(){ | |
return this.statusId; | |
} | |
}; | |
console.log(module.getStatus()); //4 | |
var getModuleStatus = module.getStatus; | |
console.log(getModuleStatus()); //this refers to global //1 | |
var boundGetStatus = getModuleStatus.bind(module); | |
console.log(boundGetStatus()); //4 |
Now we'll create bind method on the Function object's prototype, which makes bind available for all functions in our program. (More info)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var first_object = { | |
num: 4 | |
}; | |
function multi(mul) { | |
return this.num * mul; | |
} | |
Function.prototype.bind = function (obj) { | |
var method = this, | |
temp = function () { | |
return method.apply(obj, arguments); | |
}; | |
return temp; | |
} | |
var first_mul = multi.bind(first_object); | |
console.log(first_mul(5)); //20 |
When multiply.bind(obj) is called, JavaScript creates an execution context for bind method, setting this to multiply function, and setting the 1st argument obj to refer first_object.
example 2
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var first_object = { | |
a: 2, | |
b: 4 | |
}; | |
function adder(val1, val2) { | |
return this.a + this.b + val1 + val2; | |
} | |
Function.prototype.bind = function (obj) { | |
var method = this, | |
temp = function () { | |
return method.apply(obj, arguments); | |
}; | |
return temp; | |
} | |
var first = adder.bind(first_object); | |
console.log(first(5, 6)); |
The beauty of this solution is creation of method and setting it to this.
When the anonymous function is created on next line, method is accessible via its scope chain, as is obj (this couldn't be used here, because when newly created func. is executed, this will be overwritten by a new, local context) This alias to this makes it possible to use apply to execute the mulitply function, passing in obj to ensure that the context is set correctly. temp is a closure that when returned at the end of bind call, can be used in any context what so ever.
Other : Chaining .bind()
JQuery Proxy
JQuery Proxy is another way you can use to make sure this in a function will be the value you desire. It takes a function as the input and returns a new one that will always have a particular context. Read following articles Understanding jQuery proxy and usage of jQuery proxy for more information.
Other : Chaining .bind()
JQuery Proxy
JQuery Proxy is another way you can use to make sure this in a function will be the value you desire. It takes a function as the input and returns a new one that will always have a particular context. Read following articles Understanding jQuery proxy and usage of jQuery proxy for more information.
0 comments:
Post a Comment