JavaScript中的this關(guān)鍵字用法詳解
先舉一個生活例子:
小明正在跑步,他看起來很開心
這里的小明是主語,如果沒有這個主語,那么后面的代詞『他』將毫無意義。有了主語,代詞才有了可以指代的事物。
類比到JavaScript的世界中,我們在調(diào)用一個對象的方法的時候,需要先指明這個對象,再指明要調(diào)用的方法。
var xiaoming = {
name: 'Xiao Ming',
run: function() {
console.log(`${this.name} seems happy`);
},
};
xiaoming.run();在上面的例子中,第8行中的xiaoming指定了run方法運行時的主語。因此,在run中,我們才可以用this來代替xiaoming這個對象??梢钥吹?code>this起了代詞的作用。
同樣的,對于一個JavaScript類,在將它初始化之后,我們也可以用類似的方法來理解:類的實例在調(diào)用其方法的時候,將作為主語,其方法中的this就自然變成了指代主語的代詞。
class People {
constructor(name) {
// 在用new關(guān)鍵字實例化一個對象的時候,相當于在說,
// “創(chuàng)建一個People類實例(主語),它(this)的name是……”
// 所以這里的this就是新創(chuàng)建的People類實例
this.name = name;
}
run() {
console.log(`${this.name} seems happy.`)
}
}
// new關(guān)鍵字實例化一個類
var xiaoming = new People('xiaoming');
xiaoming.run();這就是我認為this關(guān)鍵字設(shè)計得精彩的地方!如果將調(diào)用方法的語句var xiaoming = new People('xiaoming');和方法本身的代碼連起來,像英語一樣讀,其實是完全通順的。
this的綁定
句子的主語是可以變的,例如在下面的場景中,run被賦值到小芳(xiaofang)身上之后,調(diào)用xiaofang.run,主語就變成了小芳!
var xiaofang = {
name: 'Xiao Fang',
};
var xiaoming = {
name: 'Xiao Ming',
run: function() {
console.log(`${this.name} seems happy`);
},
};
xiaofang.run = xiaoming.run;
// 主語變成了小芳
xiaofang.run();但是如果小明很摳門,不愿意將run方法借給小芳以后,this就變成了小芳的話,那么小明要怎么做呢?他可以通過Function.prototype.bind讓run運行時候的this永遠為小明自己。
var xiaofang = {
name: 'Xiao Fang',
};
var xiaoming = {
name: 'Xiao Ming',
run: function() {
console.log(`${this.name} seems happy`);
},
};
// 將小明的run方法綁定(bind)后,返回的還是一個
// 函數(shù),但是這個函數(shù)之后被調(diào)用的時候就算主語不是小明,
// 它的this依然是小明
xiaoming.run = xiaoming.run.bind(xiaoming);
xiaofang.run = xiaoming.run;
// 主語雖然是小芳,但是最后this還是小明
xiaofang.run();那么同一個函數(shù)被多次bind之后,到底this是哪一次bind的對象呢?你可以自己嘗試看看。
call與apply
Function.prototype.call允許你在調(diào)用一個函數(shù)的時候指定它的this的值。
var xiaoming = {
name: 'Xiao Ming'
};
function run(today, mood) {
console.log(`Today is ${today}, ${this.name} seems ${mood}`);
}
// 函數(shù)的call方法第一個參數(shù)是this的值
// 后續(xù)只需按函數(shù)參數(shù)的順序傳參即可
run.call(xiaoming, 'Monday', 'happy')Function.prototype.apply和Function.prototype.call的功能是一模一樣的,區(qū)別進在于,apply里將函數(shù)調(diào)用所需的所有參數(shù)放到一個數(shù)組當中。
var xiaoming = {
name: 'Xiao Ming'
};
function run(today, mood) {
console.log(`Today is ${today}, ${this.name} seems ${mood}`);
}
// apply只接受兩個參數(shù)
// 第二個參數(shù)是一個數(shù)組,這個數(shù)組的元素被按順序
// 作為run調(diào)用的參數(shù)
run.apply(xiaoming, ['Monday', 'happy'])那么call/apply和上面的bind混用的時候是什么樣的行為呢?這個也留給大家自行驗證。但是在一般情況下,我們應(yīng)該避免混用它們,否則會造成代碼檢查或者調(diào)試的時候難以跟蹤this的值的問題。
當方法失去主語的時候,this不再有?
其實大家可以發(fā)現(xiàn)我的用詞,當一個function被調(diào)用的時候是有主語的時候,它是一個方法;當一個function被調(diào)用的時候是沒有主語的時候,它是一個函數(shù)。當一個函數(shù)運行的時候,它雖然沒有主語,但是它的this的值會是全局對象。在瀏覽器里,那就是window。當然了,前提是函數(shù)沒有被bind過,也不是被apply或call所調(diào)用。
那么function作為函數(shù)的情景有哪些呢?
首先,全局函數(shù)的調(diào)用就是最簡單的一種。
function bar() {
console.log(this === window); // 輸出:true
}
bar();立即調(diào)用的函數(shù)表達式(IIFE,Immediately-Invoked Function Expression)也是沒有主語的,所以它被調(diào)用的時候this也是全局對象。
(function() {
console.log(this === window); // 輸出:true
})();但是,當函數(shù)被執(zhí)行在嚴格模式(strict-mode)下的時候,函數(shù)的調(diào)用時的this就是undefined了。這是很值得注意的一點。
function bar() {
'use strict';
console.log('Case 2 ' + String(this === undefined)); // 輸出:undefined
}
bar();不可見的調(diào)用
有時候,你沒有辦法看到你定義的函數(shù)是怎么被調(diào)用的。因此,你就沒有辦法知道它的主語。下面是一個用jQuery添加事件監(jiān)聽器的例子。
window.val = 'window val';
var obj = {
val: 'obj val',
foo: function() {
$('#text').bind('click', function() {
console.log(this.val);
});
}
};
obj.foo();在事件的回調(diào)函數(shù)(第6行開始定義的匿名函數(shù))里面,this的值既不是window,又不是obj,而是頁面上id為text的HTML元素。
var obj = {
foo: function() {
$('#text').bind('click', function() {
console.log(this === document.getElementById('text')); // 輸出:true
});
}
};
obj.foo();這是因為匿名函數(shù)是被jQuery內(nèi)部調(diào)用的,我們不知道它調(diào)用的時候的主語是什么,或者是否被bind等函數(shù)修改過this的值。所以,當你將匿名函數(shù)交給程序的其他部分調(diào)用的時候,需要格外地謹慎。
如果我們想要在上面的回調(diào)函數(shù)里面使用obj的val值,除了直接寫obj.val之外,還可以在foo方法中用一個新的變量that來保存foo運行時this的值。這樣說有些繞口,我們看下例子便知。
window.val = 'window val';
var obj = {
val: 'obj val',
foo: function() {
var that = this; // 保存this的引用到that,這里的this實際上就是obj
$('#text').bind('click', function() {
console.log(that.val); // 輸出:obj val
});
}
};
obj.foo();另外一種方法就是為該匿名函數(shù)bind了。
window.val = 'window val';
var obj = {
val: 'obj val',
foo: function() {
$('#text').bind('click', function() {
console.log(this.val); // 輸出:obj val
}.bind(this));
}
};
obj.foo();總結(jié)
在JavaScript中this的用法的確是千奇百怪,但是如果利用自然語言的方式來理解,一切就順理成章了。不知道你讀完這篇文章的時候理解了嗎?
以上就是JavaScript中的this關(guān)鍵字用法詳解的詳細內(nèi)容,更多關(guān)于JavaScript this關(guān)鍵字的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
JavaScript正則函數(shù)中test和match的區(qū)別解析
在javascript中,用于檢測一個字符串是否匹配某個模式用的比較多的就是test和match方法。,這篇文章主要介紹了js正則函數(shù)中test和match的區(qū)別,需要的朋友可以參考下2022-11-11
javascript的replace方法結(jié)合正則使用實例總結(jié)
這篇文章主要介紹了javascript的replace方法結(jié)合正則使用技巧,實例總結(jié)了replace方法配合正則表達式進行變量、分組、字符等替換技巧,需要的朋友可以參考下2016-06-06
IE6-IE9中tbody的innerHTML不能賦值的解決方法
這篇文章主要介紹了IE6-IE9中tbody的innerHTML不能賦值的解決方法,需要的朋友可以參考下2014-06-06
推薦幾個不錯的console調(diào)試技巧實現(xiàn)
這篇文章主要介紹了推薦幾個不錯的console調(diào)試技巧實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12
JavaScript數(shù)組Array對象增加和刪除元素方法總結(jié)
這篇文章主要介紹了JavaScript數(shù)組Array對象增加和刪除元素方法,實例總結(jié)了pop方法、push方法、splice方法、concat方法等的使用技巧,需要的朋友可以參考下2015-01-01

