簡單快速理解js中的this呢蔫、call和apply

注:本文案例環(huán)境為非嚴(yán)格模式切心,嚴(yán)格模式下禁止關(guān)鍵字this指向全局對(duì)象

一、方法是怎么執(zhí)行的?

首先說一下js中方法的執(zhí)行绽昏,在window全局下聲明一個(gè)方法a:

function a () {
  console.log(this);
}
a();//window

全局中執(zhí)行這個(gè)方法普遍的方法是直接a()扬霜,這個(gè)方法的執(zhí)行環(huán)境是window,控制臺(tái)會(huì)打印出window對(duì)象而涉。

那么為什么會(huì)打印出window對(duì)象呢?我們可以這樣理解联予,方法的執(zhí)行必須要有個(gè)直接調(diào)用者啼县,剛才那個(gè)方法a是定義在window全局下的,window下的變量和方法有個(gè)特點(diǎn)就是訪問和調(diào)用的時(shí)候可以省略window沸久!所以剛才執(zhí)行a() === window.a()季眷,也就是說,執(zhí)行a方法時(shí)的直接調(diào)用者是window卷胯。子刮!

上面有提到直接調(diào)用者,怎么看待這個(gè)直接調(diào)用者呢窑睁?舉個(gè)例子挺峡,聲明一個(gè)全局對(duì)象obj:

var name = "window-name";
var obj = {
    name:"obj-name",
    a:function(){
        console.log(this.name);
    },
    b:{
        name:"b-name",
        a:function(){
            console.log(this.name);
        }
    }
}
obj.a();//obj-name
obj.b.a();//b-name

分別執(zhí)行obj.a();和obj.b.a();控制臺(tái)會(huì)分別打印出obj-name和b-name(這里obj.a() === window.obj.a(),obj.b.a() === window.obj.b.a())担钮,方法執(zhí)行時(shí)的直接調(diào)用者就是離這個(gè)被調(diào)用方法最近的那個(gè)對(duì)象橱赠,兩個(gè)分別是obj和obj.b,打印出的name分別是obj的name和obj.b的name箫津。

二狭姨、this指向了誰?

那么函數(shù)里面的this到底是誰呢苏遥?this就是這個(gè)方法被調(diào)用時(shí)的直接調(diào)用者饼拍。可以再來個(gè)特殊的例子田炭,理解這個(gè)例子了就能很好理解this指向了誰师抄。在剛才的基礎(chǔ)上定義一個(gè)全局變量:

var ax = obj.b.a;
ax();//window-name

此時(shí)執(zhí)行ax();控制臺(tái)則會(huì)打印出window-name;為什么會(huì)打印出window-name?這是因?yàn)?ax 是定義在window全局下的變量诫肠,執(zhí)行ax()時(shí)的直接調(diào)用者是window(ax() === window.ax())司澎,所以執(zhí)行ax()時(shí)內(nèi)部的this就是它的直接調(diào)用者window,因此打印出的值就是定義在window下的name的值栋豫,所以本文最開始時(shí)的a()挤安,執(zhí)行后會(huì)打印window,因?yàn)閮?nèi)部的this指向的是a的調(diào)用者window丧鸯。

實(shí)際上在非嚴(yán)格模式下蛤铜,如果方法有直接調(diào)用者,那么this指向的是這個(gè)直接調(diào)用者,在沒有直接調(diào)用者(比如回調(diào)函數(shù))的情況下this指向的是全局對(duì)象(瀏覽器中是window围肥,node中是global)剿干。

三、call和apply改變了什么穆刻?

理解了函數(shù)的直接調(diào)用者this近范,再說call和apply就比較容易理解了。
在此對(duì)call和apply不做過多的定義性解釋鸟顺,先來看下調(diào)用了call后誰是那個(gè)被執(zhí)行的方法或粮,直接代碼示例:

function fn1 () {
    console.log(1);
};
function fn2 () {
    console.log(2);
};
fn1.call(fn2);//1

執(zhí)行fn1.call(fn2);控制臺(tái)會(huì)打印1,這里可以說明fn1調(diào)用call后被執(zhí)行的方法還是fn1朵锣。一定要弄清楚誰是這個(gè)被執(zhí)行的方法谬盐,就是調(diào)用call的函數(shù),而fn2現(xiàn)在的身份是替代window作為fn1的直接調(diào)用者诚些,這是理解call和apply的關(guān)鍵飞傀,也可以運(yùn)行下fn2.call(fn1);//2來驗(yàn)證被執(zhí)行的方法是誰。那么call的作用是什么呢诬烹?
再來個(gè)代碼示例:

var obj1 = {
    num : 20,
    fn : function(n){
        console.log(this.num+n);
    }
};
var obj2 = {
    num : 15,
    fn : function(n){
        console.log(this.num-n);
    }
};
obj1.fn.call(obj2,10);//25

執(zhí)行obj1.fn.call(obj2,10);控制臺(tái)會(huì)打印25砸烦,call在此的作用其實(shí)很簡單,就是在執(zhí)行obj1.fn的時(shí)候把這個(gè)fn的直接調(diào)用者由obj1變?yōu)閛bj2椅您,obj1.fn(n)內(nèi)部的this經(jīng)過call的作用指向了obj2外冀,所以this.num就是obj2.num,10作為執(zhí)行obj1.fn時(shí)傳入的參數(shù)掀泳,obj2.num是15雪隧,因此打印出的值是15+10=25。

所以我們可以這樣理解:call的作用是改變了那個(gè)被執(zhí)行的方法(也就是調(diào)用call的那個(gè)方法)的直接調(diào)用者员舵!而這個(gè)被執(zhí)行的方法內(nèi)部的this也會(huì)重新指向那個(gè)新的調(diào)用者脑沿,就是call方法所接收的第一個(gè)obj參數(shù)。還有兩個(gè)特殊情況就是當(dāng)這個(gè)obj參數(shù)為null或者undefined的時(shí)候马僻,this會(huì)指向window庄拇。

四、call和apply的區(qū)別

call方法除了第一個(gè)obj參數(shù)外韭邓,還接受一串參數(shù)作為被執(zhí)行的方法的參數(shù)措近,apply用法和call類似,只不過除第一個(gè)obj參數(shù)外女淑,接收的第二個(gè)參數(shù)是一個(gè)數(shù)組來作為被執(zhí)行的方法的參數(shù)瞭郑。

五、延伸拓展

我們來執(zhí)行下面的代碼:

fn1.call.call(fn2);//2

執(zhí)行fn1.call.call(fn2);控制臺(tái)會(huì)打印出2鸭你,先不說為什么會(huì)打印出2屈张,先來理解下fn1.call.call是什么擒权,call()方法是Function對(duì)象原型鏈上的方法,所以fn1這個(gè)函數(shù)可以通過原型鏈繼承使用這個(gè)方法阁谆,也就是說fn1.call === Function.prototype.call === Function.call碳抄。所以fn1.call.call(fn2) === Function.call.call(fn2),可以把Function.call先看做一個(gè)整體场绿,用FunCall來表示如下:

FunCall.call(fn2);

這樣就比較好理解剖效,就是fn2作為FunCall的直接調(diào)用者來執(zhí)行FunCall,相當(dāng)于fn2作為直接調(diào)用者執(zhí)行了FunCall()焰盗,而FunCall === Function.call贱鄙,所以就相當(dāng)于是fn2.call()。

此時(shí)call沒有傳入對(duì)象姨谷,那么全局對(duì)象window就會(huì)作為默認(rèn)對(duì)象,也就是相當(dāng)于fn2.call(window)映九,再繼續(xù)解釋就是window.fn2.call(window)梦湘,把fn2的直接調(diào)用對(duì)象由window改變成window,相當(dāng)于沒有改變fn2的直接調(diào)用對(duì)象件甥,所以就相當(dāng)于直接執(zhí)行了fn2();控制臺(tái)會(huì)打印出2捌议。

此外還有Function.call.apply和Function.apply.call等多種組合,原理都類似引有,只不過接收的參數(shù)類型不太一樣瓣颅,可以嘗試一下。加深對(duì)call和apply的理解譬正。

六宫补、補(bǔ)充bind

bind用法和call類似,只不過調(diào)用bind后方法不能立即執(zhí)行需要再次調(diào)用曾我,其實(shí)就是柯里化的一個(gè)語法糖粉怕。我們來實(shí)現(xiàn)一個(gè)簡易版的bind方法,命名為bindFn抒巢,大致就能了解bind了:

Function.prototype.bindFn = function() {
    var args = Array.prototype.slice.call(arguments);//得到傳入的參數(shù)
    var obj = args.shift();//得到第一個(gè)傳入的對(duì)象
    var self = this; // 調(diào)用bindFn的函數(shù)
    
    return function() { // return一個(gè)函數(shù) 實(shí)現(xiàn)柯里化
        //拼接新參數(shù)
        var newArgs = args.concat(Array.prototype.slice.call(arguments));
        //下面這里使用了apply,用來改變self的直接調(diào)用者
        return self.apply(obj,newArgs);
    }
}
//測試一下贫贝,doSum方法實(shí)現(xiàn)對(duì)傳入的參數(shù)的累加,并把累加結(jié)果返回
function doSum(){
    var arg = Array.prototype.slice.call(arguments);
    return arg.length ? arg.reduce((a,b) => a + b) : "";
}
var newDoSum = doSum.bindFn(null,1,2,3);
console.log(newDoSum());//6
console.log(newDoSum(4));//10
console.log(newDoSum(4,5));//15
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末蛉谜,一起剝皮案震驚了整個(gè)濱河市稚晚,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌型诚,老刑警劉巖客燕,帶你破解...
    沈念sama閱讀 216,372評(píng)論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異俺驶,居然都是意外死亡幸逆,警方通過查閱死者的電腦和手機(jī)棍辕,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來还绘,“玉大人楚昭,你說我怎么就攤上這事∨那辏” “怎么了抚太?”我有些...
    開封第一講書人閱讀 162,415評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長昔案。 經(jīng)常有香客問我尿贫,道長,這世上最難降的妖魔是什么踏揣? 我笑而不...
    開封第一講書人閱讀 58,157評(píng)論 1 292
  • 正文 為了忘掉前任庆亡,我火速辦了婚禮,結(jié)果婚禮上捞稿,老公的妹妹穿的比我還像新娘又谋。我一直安慰自己,他們只是感情好娱局,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,171評(píng)論 6 388
  • 文/花漫 我一把揭開白布彰亥。 她就那樣靜靜地躺著,像睡著了一般衰齐。 火紅的嫁衣襯著肌膚如雪任斋。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,125評(píng)論 1 297
  • 那天耻涛,我揣著相機(jī)與錄音废酷,去河邊找鬼。 笑死抹缕,一個(gè)胖子當(dāng)著我的面吹牛锦积,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播歉嗓,決...
    沈念sama閱讀 40,028評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼丰介,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了鉴分?” 一聲冷哼從身側(cè)響起哮幢,我...
    開封第一講書人閱讀 38,887評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎志珍,沒想到半個(gè)月后橙垢,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,310評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡伦糯,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,533評(píng)論 2 332
  • 正文 我和宋清朗相戀三年柜某,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了嗽元。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,690評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡喂击,死狀恐怖剂癌,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情翰绊,我是刑警寧澤佩谷,帶...
    沈念sama閱讀 35,411評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站监嗜,受9級(jí)特大地震影響谐檀,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜裁奇,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,004評(píng)論 3 325
  • 文/蒙蒙 一桐猬、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧刽肠,春花似錦课幕、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽杜秸。三九已至放仗,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間撬碟,已是汗流浹背诞挨。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評(píng)論 1 268
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留呢蛤,地道東北人惶傻。 一個(gè)月前我還...
    沈念sama閱讀 47,693評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像其障,于是被迫代替她去往敵國和親银室。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,577評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容