JavaScript中的call和apply的用途以及區(qū)別
apply 接受兩個參數(shù),第一個參數(shù)指定了函數(shù)體內(nèi)this 對象的指向,第二個參數(shù)為一個帶下標(biāo)的集合,這個集合可以為數(shù)組,也可以為類數(shù)組,apply 方法把這個集合中的元素作為參數(shù)傳遞給被調(diào)用的函數(shù):
var func = function( a, b, c ){
alert ( [ a, b, c ] ); // 輸出 [ 1, 2, 3 ]
};
func.apply( null, [ 1, 2, 3 ] );
在這段代碼中,參數(shù) 1、2、3 被放在數(shù)組中一起傳入func函數(shù),它們分別對應(yīng)func參數(shù)列表中的a、b、c。
call 傳入的參數(shù)數(shù)量不固定,跟apply 相同的是,第一個參數(shù)也是代表函數(shù)體內(nèi)的this 指向,從第二個參數(shù)開始往后,每個參數(shù)被依次傳入函數(shù):
var func = function( a, b, c ){
alert ( [ a, b, c ] ); // 輸出 [ 1, 2, 3 ]
};
func.call( null, 1, 2, 3 );
當(dāng)調(diào)用一個函數(shù)時,JavaScript 的解釋器并不會計較形參和實參在數(shù)量、類型以及順序上的區(qū)別,JavaScript 的參數(shù)在內(nèi)部就是用一個數(shù)組來表示的。從這個意義上說,apply比call的使用率更高,我們不必關(guān)心具體有多少參數(shù)被傳入函數(shù),只要用apply 一股腦地推過去就可以了。call是包裝在apply上面的一顆語法糖,如果我們明確地知道函數(shù)接受多少個參數(shù),而且想一目了然地表達(dá)形參和實參的對應(yīng)關(guān)系,那么也可以用call 來傳送參數(shù)。
call和apply的用途
1. 改變this 指向
call 和apply 最常見的用途是改變函數(shù)內(nèi)部的this 指向,我們來看個例子:
var obj1 = {
name: 'sven'
};
var obj2 = {
name: 'anne'
};
window.name = 'window';
var getName = function(){
alert ( this.name );
};
getName(); // 輸出: window
getName.call( obj1 ); // 輸出: sven
getName.call( obj2 ); // 輸出: anne
當(dāng)執(zhí)行g(shù)etName.call( obj1 )這句代碼時,getName 函數(shù)體內(nèi)的this 就指向obj1 對象,所以此處的
var getName = function(){
alert ( this.name );
};
實際上相當(dāng)于:
var getName = function(){
alert ( obj1.name ); // 輸出: sven
};
在實際開發(fā)中,經(jīng)常會遇到this指向被不經(jīng)意改變的場景,比如有一個div節(jié)點,div節(jié)點的onclick 事件中的this 本來是指向這個div的:
document.getElementById( 'div1' ).onclick = function(){
alert( this.id ); // 輸出:div1
};
假如該事件函數(shù)中有一個內(nèi)部函數(shù)func,在事件內(nèi)部調(diào)用func 函數(shù)時,func 函數(shù)體內(nèi)的this就指向了window,而不是我們預(yù)期的div,見如下代碼:
document.getElementById( 'div1' ).onclick = function(){
alert( this.id ); // 輸出:div1
var func = function(){
alert ( this.id ); // 輸出:undefined
}
func();
};
這時候我們用call 來修正func 函數(shù)內(nèi)的this,使其依然指向div:
document.getElementById( 'div1' ).onclick = function(){
var func = function(){
alert ( this.id ); // 輸出:div1
}
func.call( this );
};
2. Function.prototype.bind
大部分高級瀏覽器都實現(xiàn)了內(nèi)置的Function.prototype.bind,用來指定函數(shù)內(nèi)部的this 指向,即使沒有原生的Function.prototype.bind 實現(xiàn),我們來模擬一個也不是難事,代碼如下:
Function.prototype.bind = function( context ){
var self = this; // 保存原函數(shù)
return function(){ // 返回一個新的函數(shù)
return self.apply( context, arguments ); // 執(zhí)行新的函數(shù)的時候,會 把之前傳入的context
// 當(dāng)作新函數(shù)體內(nèi)的this
}
};
var obj = {
name: 'sven'
};
var func = function(){
alert ( this.name ); // 輸出:sven
}.bind( obj);
func();
我們通過Function.prototype.bind 來“包裝”func 函數(shù),并且傳入一個對象context 當(dāng)作參數(shù),這個context 對象就是我們想修正的this 對象。
在Function.prototype.bind 的內(nèi)部實現(xiàn)中,我們先把func 函數(shù)的引用保存起來,然后返回一個新的函數(shù)。當(dāng)我們在將來執(zhí)行func 函數(shù)時,實際上先執(zhí)行的是這個剛剛返回的新函數(shù)。在新函數(shù)內(nèi)部,self.apply( context, arguments )這句代碼才是執(zhí)行原來的func 函數(shù),并且指定context對象為func 函數(shù)體內(nèi)的this。
這是一個簡化版的Function.prototype.bind 實現(xiàn),通常我們還會把它實現(xiàn)得稍微復(fù)雜一點,
使得可以往func 函數(shù)中預(yù)先填入一些參數(shù):
Function.prototype.bind = function(){
var self = this, // 保存原函數(shù)
context = [].shift.call( arguments ), // 需要綁定的this 上下文
args = [].slice.call( arguments ); // 剩余的參數(shù)轉(zhuǎn)成數(shù)組
return function(){ // 返回一個新的函數(shù)
return self.apply( context, [].concat.call( args, [].slice.call( arguments ) ) );
// 執(zhí)行新的函數(shù)的時候,會把之前傳入的context 當(dāng)作新函數(shù)體內(nèi)的this
// 并且組合兩次分別傳入的參數(shù),作為新函數(shù)的參數(shù)
}
};
var obj = {
name: 'sven'
};
var func = function( a, b, c, d ){
alert ( this.name ); // 輸出:sven
alert ( [ a, b, c, d ] ) // 輸出:[ 1, 2, 3, 4 ]
}.bind( obj, 1, 2 );
func( 3, 4 );
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!
- javascript中apply和call方法的作用及區(qū)別說明
- js apply/call/caller/callee/bind使用方法與區(qū)別分析
- JavaScript學(xué)習(xí)點滴 call、apply的區(qū)別
- JavaScript中apply與call的用法意義及區(qū)別說明
- 在JavaScript中call()與apply()區(qū)別
- JavaScript中的this,call,apply使用及區(qū)別詳解
- 理解Javascript的caller,callee,call,apply區(qū)別
- 深入理解JavaScript中的call、apply、bind方法的區(qū)別
- javascript中apply、call和bind的使用區(qū)別
- Javascript call和apply區(qū)別及使用方法
- 深入理解關(guān)于javascript中apply()和call()方法的區(qū)別
- JavaScript中call和apply方法的區(qū)別實例分析
相關(guān)文章
JS使用面向?qū)ο蠹夹g(shù)實現(xiàn)的tab選項卡效果示例
這篇文章主要介紹了JS使用面向?qū)ο蠹夹g(shù)實現(xiàn)的tab選項卡效果,結(jié)合具體實例形式分析了js面向?qū)ο蠹夹g(shù)與tab選項卡功能的具體實現(xiàn)技巧,需要的朋友可以參考下2017-02-02
JavaScrip數(shù)組刪除特定元素的幾種方法總結(jié)
從js數(shù)組中刪除指定元素是我們每個人都遇到的問題,網(wǎng)上這方面的資料也很多,但有的時間過于久遠(yuǎn),有的內(nèi)容不夠全面,所以自己來整理下,這篇文章主要給大家總結(jié)介紹了關(guān)于JavaScrip數(shù)組刪除特定元素的多種方法,需要的朋友可以參考下。2017-09-09
學(xué)習(xí)JavaScript設(shè)計模式之代理模式
這篇文章主要為大家介紹了JavaScript設(shè)計模式中的狀態(tài)模式,對JavaScript設(shè)計模式感興趣的小伙伴們可以參考一下2016-01-01
JavaScript網(wǎng)頁表單form中禁止自動提交的兩種方式
本文是我本人在開發(fā)網(wǎng)頁時,在表單中加入了button按鈕,本來是用于jQuery點擊相應(yīng)事件的按鈕,然后,但我點擊button時,發(fā)現(xiàn)不是達(dá)到j(luò)Query指定效果,所以在本文中,我們將討論網(wǎng)頁表單(form)中提交的兩種方式,需要的朋友可以參考下2024-06-06
js獲取修改title與jQuery獲取修改title的方法
這篇文章主要介紹了js獲取修改title與jQuery獲取修改title的方法,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-02-02
JavaScript學(xué)習(xí)筆記--常用的互動方法
本文對JavaScript中常用的互動方法進(jìn)行實例分析介紹,圖文并茂,有需要的朋友可以看下2016-12-12

