JavaScript中this指向問題
記得初學 JavaScript 時,其中 this
的指向問題曾讓我頭疼不已,我還曾私自將其與閉包、原型(原型鏈)並稱 JS 武林中的三大魔頭。如果你要想在 JS 武林中稱霸一方,必須將這三大魔頭擊倒。個人認為在這三大魔頭中,this
指向問題的武功最菜(難度最低)。俗話說柿子撿軟的捏,那我們就先從 this
指向問題下手。
先記住攻克 this
指向問題的口訣(前輩們的總結):哪個物件呼叫函式,函式裡的 this
就預設指向哪個物件(注意 this
只能指向物件)。這裡說“預設指向”是因為我們通過箭頭函式、call、apply、bind等手段來改變 this
的指向。現在我們只討論 this
的預設指向。
全域性作用域下以及全域性作用域的函式中,this預設指向全域性物件window
在嚴格模式下,全域性作用域的函式中,this預設指向 undefined, 這是嚴格模式所規定的。
// 非嚴格模式下
console.log(this); // Window
function doSomething(){
console.log(this); // Window
}
doSomething(); // 這裡可以看成window.doSomething(),所以函式裡的this指向全域性物件window
// 嚴格模式下
'use strict';
console.log(this); // Window
function doInStrict(){
console.log(this); // undefined
}
doInStrict();
物件裡的函式,this指向該物件
var a = 1;
var obj = {
a: 2,
fn: function(){
console.log(this); // {a: 2, fn: ƒ}
console.log(this.a); // 2
}
};
obj.fn();
上面函式被呼叫後,從列印結果可以看出此時 this
指向的是呼叫函式的物件 obj。如果將物件中的函式賦給全域性物件中定義的變數 fn1,執行 fn1 又會出現什麼結果呢?
var a = 1;
var obj = {
a: 2,
fn: function(){
console.log(this); // Winidow
console.log(this.a); // 1
}
};
var fn1 = obj.fn;
fn1(); // 可以看成window.fn1();
從上面的例子可以看出,fn1 與 obj.fn 指向的函式是相同的,但是呼叫它的物件不同,那麼函式中 this
的指向也就不一樣了。
再看一個比較複雜的例子:
var a = 0;
function fn(){
consoloe.log(this.a);
}
var obj1 = {
a: 1,
fn: function(){
console.log(this.a);
}
};
var obj2 = {
a: 2,
fn: function(){
fn();
obj1.fn();
console.log(this.a);
}
}
obj2.fn();
先說下執行結果,分別列印 0 1 2
。當 obj2 呼叫 fn 函式時,先執行的是 fn()
,這個函式是在全域性作用域中定義的,該呼叫可以看成 window.fn()
,所以,該函式內部的 this
指向的是 window 全域性物件,this.a
自然就是全域性物件中的 a 值(0)。
接著執行的是 obj1.fn()
,它會從 obj1 中找到 fn 函式並執行。obj1 中的函式 fn 執行時呼叫它的物件是 obj1,所以,此時函式內部的 this
指向的就是 obj1 自身。那麼 this.a
查到的值也就是物件 obj1 中 a 的值(1)。
最後列印函式中 this
所處的函式 fn 是被 obj2 呼叫的,那麼自然而然 this
就指向了 obj2,所以 this.a
的結果就是 2 了。
從上面這個例子我們可以看出:函式內部 this 指向跟呼叫函式的物件有關,跟函式在哪裡呼叫沒有關係。
Window內建函式的回撥函式中,this指向Window物件。
window 的內建函式( setInterval setTimeout 等),其回撥函式中的 this
指向的是window物件。
var name = 'window';
var obj = {
name: 'obj',
func: function(){
setTimeout(function () {
console.log(this.name) // window
},1000)
}
}
obj.func()
但是一般在開發中,很多場景都需要改變 this
的指向。 後面我會專門寫一篇關於更改 this
指向的文章,這裡就不再贅述了。