您好,登錄后才能下訂單哦!
今天小編給大家分享一下JavaScript函數(shù)執(zhí)行上下文的this怎么調(diào)用的相關(guān)知識(shí)點(diǎn),內(nèi)容詳細(xì),邏輯清晰,相信大部分人都還太了解這方面的知識(shí),所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。
關(guān)于 this,我們得先從執(zhí)行上下文說起。我們知道:執(zhí)行上下文中包含了變量環(huán)境、詞法環(huán)境、外部環(huán)境,當(dāng)然也包括 this,具體你可以參考下圖:
從圖中可以看出,this 是和執(zhí)行上下文綁定的,也就是說每個(gè)執(zhí)行上下文中都有一個(gè) this。執(zhí)行上下文主要分為三種——全局執(zhí)行上下文、函數(shù)執(zhí)行上下文和 eval 執(zhí)行上下文,所以對(duì)應(yīng)的 this 也只有這三種——全局執(zhí)行上下文中的 this、函數(shù)中的 this 和 eval 中的 this。
不過由于 eval 我們使用的不多,所以本文我們對(duì)此就不做介紹了,如果你感興趣的話,可以自行搜索和學(xué)習(xí)相關(guān)知識(shí)。
那么接下來我們就重點(diǎn)講解下全局執(zhí)行上下文中的 this和函數(shù)執(zhí)行上下文中的 this。
首先我們來看看全局執(zhí)行上下文中的 this 是什么。
你可以在控制臺(tái)中輸入console.log(this)
來打印出來全局執(zhí)行上下文中的 this,最終輸出的是 window 對(duì)象。所以你可以得出這樣一個(gè)結(jié)論:全局執(zhí)行上下文中的 this 是指向 window 對(duì)象的。這也是 this 和作用域鏈的唯一交點(diǎn),作用域鏈的最底端包含了 window 對(duì)象,全局執(zhí)行上下文中的 this 也是指向 window 對(duì)象。
現(xiàn)在你已經(jīng)知道全局對(duì)象中的 this 是指向 window 對(duì)象了,那么接下來,我們就來重點(diǎn)分析函數(shù)執(zhí)行上下文中的 this。還是先看下面這段代碼:
function foo() { console.log(this); } foo();
我們?cè)?foo 函數(shù)內(nèi)部打印出來 this 值,執(zhí)行這段代碼,打印出來的也是 window 對(duì)象,這說明在默認(rèn)情況下調(diào)用一個(gè)函數(shù),其執(zhí)行上下文中的 this 也是指向 window 對(duì)象的。估計(jì)你會(huì)好奇,那能不能設(shè)置執(zhí)行上下文中的 this 來指向其他對(duì)象呢?答案是肯定的。通常情況下,有下面三種方式來設(shè)置函數(shù)執(zhí)行上下文中的 this 值。
你可以通過函數(shù)的call方法來設(shè)置函數(shù)執(zhí)行上下文的 this 指向,比如下面這段代碼,我們就并沒有直接調(diào)用 foo 函數(shù),而是調(diào)用了 foo 的 call 方法,并將 bar 對(duì)象作為 call 方法的參數(shù)。
let bar = { myName: " name1 ", test1: 1, }; function foo() { this.myName = " name2 "; } foo.call(bar); console.log(bar); console.log(myName);
執(zhí)行這段代碼,然后觀察輸出結(jié)果,你就能發(fā)現(xiàn) foo 函數(shù)內(nèi)部的 this 已經(jīng)指向了 bar 對(duì)象,因?yàn)橥ㄟ^打印 bar 對(duì)象,可以看出 bar 的 myName 屬性已經(jīng)由“name1”變?yōu)椤皀ame2”了,同時(shí)在全局執(zhí)行上下文中打印 myName,JavaScript 引擎提示該變量未定義。
其實(shí)除了 call 方法,你還可以使用bind和apply方法來設(shè)置函數(shù)執(zhí)行上下文中的 this,僅僅是語法稍有不同。
要改變函數(shù)執(zhí)行上下文中的 this 指向,除了通過函數(shù)的 call 方法來實(shí)現(xiàn)外,還可以通過對(duì)象調(diào)用的方式,比如下面這段代碼:
var myObj = { name: " name ", showThis: function () { console.log(this); }, }; myObj.showThis();
在這段代碼中,我們定義了一個(gè) myObj 對(duì)象,該對(duì)象是由一個(gè) name 屬性和一個(gè) showThis 方法組成的,然后再通過 myObj 對(duì)象來調(diào)用 showThis 方法。執(zhí)行這段代碼,你可以看到,最終輸出的 this 值是指向 myObj 的。
所以,你可以得出這樣的結(jié)論:使用對(duì)象來調(diào)用其內(nèi)部的一個(gè)方法,該方法的 this 是指向?qū)ο蟊旧淼摹?/p>
其實(shí),你也可以認(rèn)為 JavaScript 引擎在執(zhí)行myObject.showThis()
時(shí),將其轉(zhuǎn)化為了:
myObj.showThis.call(myObj)
接下來我們稍微改變下調(diào)用方式,把 showThis 賦給一個(gè)全局對(duì)象,然后再調(diào)用該對(duì)象,代碼如下所示:
var myObj = { name: " time ", showThis: function () { this.name = " bang "; console.log(this); }, }; var foo = myObj.showThis; foo();
執(zhí)行這段代碼,你會(huì)發(fā)現(xiàn) this 又指向了全局 window 對(duì)象。
所以通過以上兩個(gè)例子的對(duì)比,你可以得出下面這樣兩個(gè)結(jié)論:
在全局環(huán)境中調(diào)用一個(gè)函數(shù),函數(shù)內(nèi)部的 this 指向的是全局變量 window。
通過一個(gè)對(duì)象來調(diào)用其內(nèi)部的一個(gè)方法,該方法的執(zhí)行上下文中的 this 指向?qū)ο蟊旧怼?/p>
你可以像這樣設(shè)置構(gòu)造函數(shù)中的 this,如下面的示例代碼:
function CreateObj() { this.name = " time "; } var myObj = new CreateObj();
在這段代碼中,我們使用 new 創(chuàng)建了對(duì)象 myObj,那你知道此時(shí)的構(gòu)造函數(shù) CreateObj 中的 this 到底指向了誰嗎?
其實(shí),當(dāng)執(zhí)行 new CreateObj() 的時(shí)候,JavaScript 引擎做了如下四件事:
首先創(chuàng)建了一個(gè)空對(duì)象 tempObj;
接著調(diào)用 CreateObj.call 方法,并將 tempObj 作為 call 方法的參數(shù),這樣當(dāng) CreateObj 的執(zhí)行上下文創(chuàng)建時(shí),它的 this 就指向了 tempObj 對(duì)象;
然后執(zhí)行 CreateObj 函數(shù),此時(shí)的 CreateObj 函數(shù)執(zhí)行上下文中的 this 指向了 tempObj 對(duì)象;
最后返回 tempObj 對(duì)象。
這樣,我們就通過 new 關(guān)鍵字構(gòu)建好了一個(gè)新對(duì)象,并且構(gòu)造函數(shù)中的 this 其實(shí)就是新對(duì)象本身。
就我個(gè)人而言,this 并不是一個(gè)很好的設(shè)計(jì),因?yàn)樗暮芏嗍褂梅椒ǘ紱_擊人的直覺,在使用過程中存在著非常多的坑。下面咱們就來一起看看那些 this 設(shè)計(jì)缺陷。
我認(rèn)為這是一個(gè)嚴(yán)重的設(shè)計(jì)錯(cuò)誤,并影響了很多開發(fā)者。
至于如何解決?你可以在函數(shù)中聲明一個(gè)變量 self 用來保存 this。當(dāng)然也可以使用 ES6 中的箭頭函數(shù)來解決這個(gè)問題。
上面我們已經(jīng)介紹過了,在默認(rèn)情況下調(diào)用一個(gè)函數(shù),其執(zhí)行上下文中的 this 是默認(rèn)指向全局對(duì)象 window 的。
不過這個(gè)設(shè)計(jì)也是一種缺陷,因?yàn)樵趯?shí)際工作中,我們并不希望函數(shù)執(zhí)行上下文中的 this 默認(rèn)指向全局對(duì)象,因?yàn)檫@樣會(huì)打破數(shù)據(jù)的邊界,造成一些誤操作。如果要讓函數(shù)執(zhí)行上下文中的 this 指向某個(gè)對(duì)象,最好的方式是通過 call 方法來顯示調(diào)用。
這個(gè)問題可以通過設(shè)置 JavaScript 的“嚴(yán)格模式”來解決。在嚴(yán)格模式下,默認(rèn)執(zhí)行一個(gè)函數(shù),其函數(shù)的執(zhí)行上下文中的 this 值是 undefined,這就解決上面的問題了。
以上就是“JavaScript函數(shù)執(zhí)行上下文的this怎么調(diào)用”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會(huì)為大家更新不同的知識(shí),如果還想學(xué)習(xí)更多的知識(shí),請(qǐng)關(guān)注億速云行業(yè)資訊頻道。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。