加深對(duì) JavaScript This 的理解

歡迎來(lái)我的博客閱讀:《加深對(duì) JavaScript This 的理解》

我相信你已經(jīng)看過(guò)很多關(guān)于 JavaScript 的 this 的談?wù)摿耸G纾热荒泓c(diǎn)進(jìn)來(lái)了,不妨繼續(xù)看下去浪感,看是否能幫你加深對(duì) this 的理解纲仍。

最近在看 《You Dont Know JS》 這本書,不得感嘆差导,就算用了 JS 很多年的老前端來(lái)看這本書被啼,我覺(jué)得還是會(huì)有不少的收獲帜消。

其中關(guān)于 this 的講解,更是加深了我對(duì) this 的理解浓体,故整理知識(shí)點(diǎn)券犁,再加上自身的理解,以自己的語(yǔ)言來(lái)描述汹碱。
對(duì)讀者來(lái)說(shuō)粘衬,算是二手知識(shí),這本書是開源的咳促,可以到本書的 Github 項(xiàng)目地址學(xué)習(xí)一手的知識(shí)稚新。

首先有一句大家都明白的話,我還是要強(qiáng)調(diào)一遍:
this 是在函數(shù)被調(diào)用時(shí)發(fā)生的綁定跪腹,它指向什么完全取決于函數(shù)在哪里被調(diào)用褂删。」

這句話很重要冲茸,這是理解 this 原理的基礎(chǔ)屯阀。
而在講解 this 之前,先要理解一下作用域的相關(guān)概念轴术。

「詞法作用域」與「動(dòng)態(tài)作用域」

通常來(lái)說(shuō)难衰,作用域一共有兩種主要的工作模型。

  • 詞法作用域
  • 動(dòng)態(tài)作用域

詞法作用域是大多數(shù)編程語(yǔ)言所采用的模式逗栽,而動(dòng)態(tài)作用域仍有一些編程語(yǔ)言在用盖袭,例如 Bash 腳本。
而 JavaScript 就是采用的詞法作用域彼宠,也就是在編程階段鳄虱,作用域就已經(jīng)明確下來(lái)了。

思考下面代碼:

function foo(){
  console.log(a);   // 輸出 2
}

function bar(){
  let a = 3;
  foo();
}

let a = 2;

bar()

因?yàn)?JavaScript 所用的是詞法作用域凭峡,自然 foo() 聲明的階段拙已,就已經(jīng)確定了變量 a 的作用域了。

倘若摧冀,JavaScript 是采用的動(dòng)態(tài)作用域倍踪,foo() 中打印的將是 3

function foo(){
  console.log(a);   // 輸出 3 (不是 2)
}

function bar(){
  let a = 3;
  foo();
}

let a = 2;

bar()

而 JavaScript 的 this 機(jī)制跟動(dòng)態(tài)作用域很相似霉涨,是在運(yùn)行時(shí)在被調(diào)用的地方動(dòng)態(tài)綁定的。

this 的四種綁定規(guī)則

在 JavaScript 中惭适,影響 this 指向的綁定規(guī)則有四種:

  • 默認(rèn)綁定
  • 隱式綁定
  • 顯式綁定
  • new 綁定

默認(rèn)綁定

這是最直接的一種方式,就是不加任何的修飾符直接調(diào)用函數(shù)楼镐,如:

function foo() {
  console.log(this.a)   // 輸出 a
}

var a = 2;  //  變量聲明到全局對(duì)象中

foo();

使用 var 聲明的變量 a癞志,被綁定到全局對(duì)象中,如果是瀏覽器框产,則是在 window 對(duì)象凄杯。
foo() 調(diào)用時(shí),引用了默認(rèn)綁定秉宿,this 指向了全局對(duì)象戒突。

隱式綁定

這種情況會(huì)發(fā)生在調(diào)用位置存在「上下文對(duì)象」的情況,如:

function foo() {
  console.log(this.a);
}

let obj1 = {
  a: 1,
  foo,
};

let obj2 = {
  a: 2,
  foo,
}

obj1.foo();   // 輸出 1
obj2.foo();   // 輸出 2

當(dāng)函數(shù)調(diào)用的時(shí)候描睦,擁有上下文對(duì)象的時(shí)候膊存,this 會(huì)被綁定到該上下文對(duì)象。
正如上面的代碼忱叭,
obj1.foo() 被調(diào)用時(shí)隔崎,this 綁定到了 obj1,
obj2.foo() 被調(diào)用時(shí),this 綁定到了 obj2韵丑。

顯式綁定

這種就是使用 Function.prototype 中的三個(gè)方法 call(), apply(), bind() 了爵卒。
這三個(gè)函數(shù),都可以改變函數(shù)的 this 指向到指定的對(duì)象撵彻,
不同之處在于钓株,call()apply() 是立即執(zhí)行函數(shù),并且接受的參數(shù)的形式不同:

  • call(this, arg1, arg2, ...)
  • apply(this, [arg1, arg2, ...])

bind() 則是創(chuàng)建一個(gè)新的包裝函數(shù)陌僵,并且返回轴合,而不是立刻執(zhí)行。

  • bind(this, arg1, arg2, ...)

apply() 接收參數(shù)的形式碗短,有助于函數(shù)嵌套函數(shù)的時(shí)候值桩,把 arguments 變量傳遞到下一層函數(shù)中。

思考下面代碼:

function foo() {
  console.log(this.a);  // 輸出 1
  bar.apply({a: 2}, arguments);
}

function bar(b) {
  console.log(this.a + b);  // 輸出 5
}

var a = 1;
foo(3);

上面代碼中豪椿, foo() 內(nèi)部的 this 遵循默認(rèn)綁定規(guī)則奔坟,綁定到全局變量中。
bar() 在調(diào)用的時(shí)候搭盾,調(diào)用了 apply() 函數(shù)咳秉,把 this 綁定到了一個(gè)新的對(duì)象中 {a: 2},而且原封不動(dòng)的接收 foo() 接收的函數(shù)鸯隅。

new 綁定

最后一種澜建,則是使用 new 操作符會(huì)產(chǎn)生 this 的綁定向挖。
在理解 new 操作符對(duì) this 的影響,首先要理解 new 的原理炕舵。
在 JavaScript 中何之,new 操作符并不像其他面向?qū)ο蟮恼Z(yǔ)言一樣,而是一種模擬出來(lái)的機(jī)制咽筋。
在 JavaScript 中溶推,所有的函數(shù)都可以被 new 調(diào)用,這時(shí)候這個(gè)函數(shù)一般會(huì)被稱為「構(gòu)造函數(shù)」奸攻,實(shí)際上并不存在所謂「構(gòu)造函數(shù)」蒜危,更確切的理解應(yīng)該是對(duì)于函數(shù)的「構(gòu)造調(diào)用」。

使用 new 來(lái)調(diào)用函數(shù)睹耐,會(huì)自動(dòng)執(zhí)行下面操作:

  1. 創(chuàng)建一個(gè)全新的對(duì)象辐赞。
  2. 這個(gè)新對(duì)象會(huì)被執(zhí)行 [[Prototype]] 連接。
  3. 這個(gè)新對(duì)象會(huì)綁定到函數(shù)調(diào)用的 this硝训。
  4. 如果函數(shù)沒(méi)有返回其他對(duì)象响委,那么 new 表達(dá)式中的函數(shù)調(diào)用會(huì)自動(dòng)返回這個(gè)新對(duì)象。

所以如果 new 是一個(gè)函數(shù)的話窖梁,會(huì)是這樣子的:

function New(Constructor, ...args){
    let obj = {};   // 創(chuàng)建一個(gè)新對(duì)象
    Object.setPrototypeOf(obj, Constructor.prototype);  // 連接新對(duì)象與函數(shù)的原型
    return Constructor.apply(obj, args) || obj;   // 執(zhí)行函數(shù)晃酒,改變 this 指向新的對(duì)象
}

function Foo(a){
    this.a = a;
}

New(Foo, 1);  // Foo { a: 1 }

所以,在使用 new 來(lái)調(diào)用函數(shù)時(shí)候窄绒,我們會(huì)構(gòu)造一個(gè)新對(duì)象并把它綁定到函數(shù)調(diào)用中的 this 上贝次。

優(yōu)先級(jí)

如果一個(gè)位置發(fā)生了多條改變 this 的規(guī)則,那么優(yōu)先級(jí)是如何的呢彰导?

看幾段代碼:

// 顯式綁定 > 隱式綁定
function foo() {
    console.log(this.a);
}

let obj1 = {
    a: 2,
    foo,
}

obj1.foo();     // 輸出 2
obj1.foo.call({a: 1});      // 輸出 1

這說(shuō)明「顯式綁定」的優(yōu)先級(jí)大于「隱式綁定」

// new 綁定 > 顯式綁定
function foo(a) {
    this.a = a;
}

let obj1 = {};

let bar = foo.bind(obj1);
bar(2);
console.log(obj1); // 輸出 {a:2}

let obj2 = new bar(3);
console.log(obj1); // 輸出 {a:2}
console.log(obj2); // 輸出 foo { a: 3 }

這說(shuō)明「new 綁定」的優(yōu)先級(jí)大于「顯式綁定」
而「默認(rèn)綁定」蛔翅,毫無(wú)疑問(wèn)是優(yōu)先級(jí)最低的。
所以優(yōu)先級(jí)順序?yàn)椋?/p>

「new 綁定」 > 「顯式綁定」 > 「隱式綁定」 > 「默認(rèn)綁定位谋∩轿觯」

所以,this 到底是什么

this 并不是在編寫的時(shí)候綁定的掏父,而是在運(yùn)行時(shí)綁定的笋轨。它的上下文取決于函數(shù)調(diào)用時(shí)的各種條件。
this 的綁定和函數(shù)聲明的位置沒(méi)有任何關(guān)系赊淑,只取決于函數(shù)的調(diào)用方式爵政。
當(dāng)一個(gè)函數(shù)被調(diào)用時(shí),會(huì)創(chuàng)建一個(gè)「執(zhí)行上下文」陶缺,這個(gè)上下文會(huì)包含函數(shù)在哪里被調(diào)用(調(diào)用棧)钾挟、函數(shù)的調(diào)用方式、傳入的參數(shù)等信息饱岸。this 就是這個(gè)記錄的一個(gè)屬性掺出,會(huì)在函數(shù)執(zhí)行的過(guò)程中用到徽千。

參考

《You Dont Know JS》- this & Object Prototypes

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市汤锨,隨后出現(xiàn)的幾起案子双抽,更是在濱河造成了極大的恐慌,老刑警劉巖闲礼,帶你破解...
    沈念sama閱讀 219,427評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件牍汹,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡位仁,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門方椎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)聂抢,“玉大人,你說(shuō)我怎么就攤上這事棠众×帐瑁” “怎么了?”我有些...
    開封第一講書人閱讀 165,747評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵闸拿,是天一觀的道長(zhǎng)空盼。 經(jīng)常有香客問(wèn)我,道長(zhǎng)新荤,這世上最難降的妖魔是什么揽趾? 我笑而不...
    開封第一講書人閱讀 58,939評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮苛骨,結(jié)果婚禮上篱瞎,老公的妹妹穿的比我還像新娘。我一直安慰自己痒芝,他們只是感情好俐筋,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著严衬,像睡著了一般澄者。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上请琳,一...
    開封第一講書人閱讀 51,737評(píng)論 1 305
  • 那天粱挡,我揣著相機(jī)與錄音,去河邊找鬼俄精。 笑死抱怔,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的嘀倒。 我是一名探鬼主播屈留,決...
    沈念sama閱讀 40,448評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼局冰,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了灌危?” 一聲冷哼從身側(cè)響起康二,我...
    開封第一講書人閱讀 39,352評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎勇蝙,沒(méi)想到半個(gè)月后沫勿,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,834評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡味混,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評(píng)論 3 338
  • 正文 我和宋清朗相戀三年产雹,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片翁锡。...
    茶點(diǎn)故事閱讀 40,133評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡蔓挖,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出馆衔,到底是詐尸還是另有隱情瘟判,我是刑警寧澤,帶...
    沈念sama閱讀 35,815評(píng)論 5 346
  • 正文 年R本政府宣布角溃,位于F島的核電站拷获,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏减细。R本人自食惡果不足惜匆瓜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望未蝌。 院中可真熱鬧陕壹,春花似錦、人聲如沸树埠。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,022評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)怎憋。三九已至又碌,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間绊袋,已是汗流浹背毕匀。 一陣腳步聲響...
    開封第一講書人閱讀 33,147評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留癌别,地道東北人皂岔。 一個(gè)月前我還...
    沈念sama閱讀 48,398評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像展姐,于是被迫代替她去往敵國(guó)和親躁垛。 傳聞我的和親對(duì)象是個(gè)殘疾皇子剖毯,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評(píng)論 2 355

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

  • 官方中文版原文鏈接 感謝社區(qū)中各位的大力支持逊谋,譯者再次奉上一點(diǎn)點(diǎn)福利:阿里云產(chǎn)品券,享受所有官網(wǎng)優(yōu)惠土铺,并抽取幸運(yùn)大...
    HetfieldJoe閱讀 6,926評(píng)論 15 54
  • 特別說(shuō)明悲敷,為便于查閱究恤,文章轉(zhuǎn)自https://github.com/getify/You-Dont-Know-JS...
    殺破狼real閱讀 693評(píng)論 0 1
  • 1. this之謎 在JavaScript中,this是當(dāng)前執(zhí)行函數(shù)的上下文后德。因?yàn)镴avaScript有4種不同的...
    百里少龍閱讀 1,005評(píng)論 0 3
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理部宿,服務(wù)發(fā)現(xiàn),斷路器探遵,智...
    卡卡羅2017閱讀 134,672評(píng)論 18 139
  • 《你不知道的JavaScript》系列叢書給出了很多顛覆以往對(duì)JavaScript認(rèn)知的點(diǎn), 讀完上卷窟赏,受益匪淺妓柜,...
    牧云云閱讀 1,165評(píng)論 0 18