日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

JavaScript中的執(zhí)行環(huán)境和作用域鏈

瀏覽:176日期:2023-10-15 10:40:23

前言

JS 中的執(zhí)行環(huán)境和作用域鏈?zhǔn)欠浅V匾母拍睿鼈兪?JS 引擎在處理 JS 代碼的時候?qū)ψ兞亢秃瘮?shù)的處理方式,這兩個概念的正確理解能夠幫助我們更好地理解和預(yù)測代碼的行為。

執(zhí)行環(huán)境

執(zhí)行環(huán)境定義了變量或者函數(shù)有權(quán)訪問的數(shù)據(jù)集合,每一個執(zhí)行環(huán)境都有一個與之關(guān)聯(lián)的變量對象,該執(zhí)行環(huán)境中定義的所有變量和函數(shù)都保存在這個對象中。我們無法直接訪問這個對象,這個對象只是在解析器處理數(shù)據(jù)的時候使用。

我們平時說的全局變量就是在最外圍的一個執(zhí)行環(huán)境中定義的變量,全局執(zhí)行環(huán)境根據(jù) ECMAScript 的不同實現(xiàn)而有不同的表示,在 Web 瀏覽器中,全局執(zhí)行環(huán)境就是 window 對象,所有的全局變量和函數(shù)就是作為 window 對象的屬性和方法創(chuàng)建的。在 nodejs 的實現(xiàn)中,全局執(zhí)行環(huán)境就是global 對象。除了全局執(zhí)行環(huán)境,每個函數(shù)都有自己的執(zhí)行環(huán)境,當(dāng)執(zhí)行流進(jìn)入一個函數(shù)時,函數(shù)的環(huán)境就會被推入一個環(huán)境棧中,而函數(shù)執(zhí)行之后,棧將其環(huán)境彈出,把控制權(quán)返回給之前的執(zhí)行環(huán)境。也就是說某個執(zhí)行環(huán)境中的代碼全部執(zhí)行完畢之后,該環(huán)境就被銷毀,保存在其中的所有變量和函數(shù)定義也隨之銷毀,全局執(zhí)行環(huán)境直到應(yīng)用程序額推出——例如網(wǎng)頁或瀏覽器被關(guān)閉時才被銷毀。

作用域鏈

前面說到每個執(zhí)行環(huán)境都有一個變量對象來保存環(huán)境中定義的變量和函數(shù),環(huán)境是層層嵌套的,所以當(dāng)代碼進(jìn)入到一個新的環(huán)境開始執(zhí)行時,會創(chuàng)建變量對象的一個作用域鏈,把嵌套的執(zhí)行環(huán)境之間的變量對象做一個有序的聯(lián)系。作用域鏈最主要的作用是確保當(dāng)前執(zhí)行環(huán)境有權(quán)訪問的變量和函數(shù),并且有序地查找。在作用域鏈的最前端始終是當(dāng)前正在執(zhí)行的代碼所處的執(zhí)行環(huán)境的變量對象,如果這個環(huán)境是一個函數(shù),就把函數(shù)的活動對象作為其變量對象,在函數(shù)中沒有定義新的變量時,這個活動獨享就是函數(shù)的 arguments 對象。作用域鏈的下一個變量都西昂來自于當(dāng)前執(zhí)行環(huán)境的包含環(huán)境,依次類推,逐層嵌套,知道全局執(zhí)行環(huán)境;全局執(zhí)行環(huán)境的變量對象始終都是作用域鏈中的最后一個都對象。

當(dāng)我們的代碼在執(zhí)行的時候,遇到的每一個標(biāo)識符解析都會沿著作用域鏈一級一級地進(jìn)行搜索,從作用域鏈的前端(當(dāng)前執(zhí)行環(huán)境的變量對象)逐級向后回溯,知道找到標(biāo)識符為止,如果在作用域鏈上沒有找到這個標(biāo)識符,通常會導(dǎo)致錯誤。我們經(jīng)常遇到的 Uncaught ReferenceError: x is not defined 就是這個錯誤在瀏覽器中的表現(xiàn)。

JS解釋器在執(zhí)行時會將變量和函數(shù)進(jìn)行聲明提前,在聲明函數(shù)的時候,會給函數(shù)一個 [[scope]] 屬性,這個屬性中包含了當(dāng)前函數(shù)所有包含環(huán)境的變量對象,也就是我們的函數(shù)在聲明提前的時候就已經(jīng)生成了他的包含環(huán)境的作用域鏈了,然后當(dāng)函數(shù)執(zhí)行的時候會把自己的 arguments 和內(nèi)部定義的函數(shù)和變量打包成一個變量對象加到 scope chain 的最后。

函數(shù)參數(shù)也被當(dāng)做變量來對待,因此起訪問規(guī)則與執(zhí)行環(huán)境中的其他變量相同。

作用域鏈的這種特性理解起來其實也是比較直觀的,但是在實際的代碼中由于情況非常多,有時候有些行為還是比較反直覺或者說容易產(chǎn)生誤解的。比如下面的情況:

作用域鏈看的是函數(shù)定義的位置而不是執(zhí)行的位置

var x = 10bar()function foo() { console.log(x)}function bar(){ var x = 30 foo()}

在這個例子里面,可能會有人誤以為 bar() 會輸出 30,我們只要理解函數(shù)其實是保存在堆中,我們給函數(shù)命名只是一個指向函數(shù)堆中地址的一個引用,當(dāng)我們執(zhí)行函數(shù)的時候根據(jù)這個引用去堆中找對應(yīng)的函數(shù)執(zhí)行。所以無論我們在哪里執(zhí)行函數(shù),函數(shù)的位置都是不變的,我們看作用域鏈也是,我們確定作用域鏈不是看函數(shù)是在哪里執(zhí)行,而是要看函數(shù)是在哪里定義,作用域鏈可以認(rèn)為是函數(shù)聲明時就已經(jīng)生成了。

個人認(rèn)為 ECMAScript 這樣處理作用域鏈?zhǔn)菫榱俗饔糜蜴溎軌虮3植蛔兌挥靡恢本S護(hù),并且根據(jù)環(huán)境的嵌套保持一致性。

閉包

除了全局執(zhí)行環(huán)境的變量對象是始終存在的,其他局部函數(shù)的變量對象都只在函數(shù)的執(zhí)行過程中存在,一般來講,函數(shù)執(zhí)行完畢之后,局部活動對象就被銷毀了,內(nèi)存中僅僅保存全局執(zhí)行環(huán)境的變量對象,但是閉包的情況是不同的。

閉包指的是有權(quán)訪問另一個函數(shù)作用域中的變量的函數(shù),比如下面這樣:

function outer(){ var scope = 'outer'; return function (){ return scope; } }var fn = outer();fn();

在一個函數(shù)內(nèi)部定義的函數(shù)會將包含函數(shù)(即外部函數(shù))的活動對象添加到他的作用域鏈中,因此在 outer 函數(shù)內(nèi)部定義的匿名函數(shù)(我們下面把這個匿名函數(shù)稱為 inner 函數(shù))的作用域鏈中,實際上會包含外部函數(shù) outer() 的活動對象,下圖可以看書當(dāng)代碼執(zhí)行時,outer 和 inner 函數(shù)的作用域鏈。

JavaScript中的執(zhí)行環(huán)境和作用域鏈

當(dāng)匿名函數(shù)從 outer() 中被返回后,inner() 函數(shù)仍然可以訪問在 outer() 中定義的所有變量,也就是說,當(dāng) outer() 函數(shù)執(zhí)行完畢后,其活動對象也不會被銷毀,因為匿名函數(shù)的作用域鏈依然在引用這個活動對象。換句話說,當(dāng) outer() 函數(shù)執(zhí)行完畢返回后,其執(zhí)行環(huán)境和作用域鏈都被銷毀,但它的活動對象依然保存在內(nèi)存中,如果匿名函數(shù)不銷毀,則這個活動對象會一直存在于內(nèi)存中。

js中的對象都是保存在堆中,我們在代碼中寫的都是對對象的引用,作用域鏈中也是,所以上面說的 outer() 函數(shù)執(zhí)行完畢后作用域鏈被銷毀但是對象還存在,其實銷毀的只是引用, js 中的垃圾處理機(jī)制的一種策略是引用計數(shù),當(dāng)某個變量或?qū)ο蟮囊么螖?shù)為 0 的時候內(nèi)存會被收回。outer 函數(shù)的變量對象的引用有兩個一個是 outer 的作用域鏈和匿名函數(shù)的作用域鏈,所以只要匿名函數(shù)不被銷毀,這個引用就一直存在,outer() 的活動對象也會一直存在。

輪子哥在知乎給過一個比較容易理解的說法:“閉”的意思不是封閉內(nèi)部狀態(tài),而是封閉外部狀態(tài),一個函數(shù)如何能夠封閉外部狀態(tài)呢,當(dāng)外部狀態(tài)的 scope 失效的時候,它自己還保留了一份。

JavaScript中的執(zhí)行環(huán)境和作用域鏈

由于閉包會攜帶包含它的函數(shù)的作用域,因此回避其他函數(shù)占用更多的內(nèi)存。過度使用閉包可能會導(dǎo)致內(nèi)存占用過多,只在必要的時候使用閉包。

總結(jié)

任何一種編程語言都有作用域的概念,我們的程序是圍繞著變量操作的,那么在設(shè)計語言的時候,變量如何儲存,儲存到哪里,我們的程序如何找到對應(yīng)的變量就是一個首先要解決的問題。而作用域就是語言設(shè)計者針對這個問題編寫的一套設(shè)計良好的規(guī)則來存儲并搜索對象,這就是作用域的概念。而 JS 中的這個規(guī)則就是作用域鏈,我們在編寫程序的時候也需要知道我們的變量(以及函數(shù))是如何儲存,以及 JS 引擎在遇到標(biāo)識符解析的時候是按照什么規(guī)則來搜索變量或者函數(shù)的,只有這樣我們才能寫出更可靠的代碼。

以上就是JavaScript中的執(zhí)行環(huán)境和作用域鏈的詳細(xì)內(nèi)容,更多關(guān)于JavaScript執(zhí)行環(huán)境和作用域鏈的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: JavaScript
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
伊人久久大香线蕉av不卡| 国产日韩三级| 国产免费久久| 日韩精品91亚洲二区在线观看| 亚洲日本国产| 欧美日韩a区| 国产精品亚洲欧美日韩一区在线 | 天堂va在线高清一区| 天海翼亚洲一区二区三区| 日本一区二区三区视频在线看| 免费观看日韩电影| 婷婷精品在线观看| 天堂va欧美ⅴa亚洲va一国产| 石原莉奈在线亚洲二区| 日韩精品一区二区三区免费视频| 国产精品久久国产愉拍| 国内亚洲精品| 久久av在线| 欧美日韩亚洲一区三区| 成人免费一区| 亚洲在线成人| 一区二区国产在线| 国产精品白丝久久av网站| 91看片一区| 婷婷五月色综合香五月| 久久九九精品| 国产视频亚洲| 欧美黄色一区| 国产第一亚洲| 中文字幕日韩亚洲| 一区二区精品伦理...| 国产日韩三级| 成人国产精品| 亚洲天堂免费| 日韩成人亚洲| 国产亚洲欧美日韩精品一区二区三区| 久久精品动漫| 美女视频黄免费的久久| 久久久精品久久久久久96| 亚洲a成人v| 99久久久久久中文字幕一区| 日韩高清中文字幕一区| 视频小说一区二区| 国产亚洲久久| 久久xxxx| 五月综合激情| 精品一区亚洲| 欧美日韩国产探花| 日韩精品永久网址| 精品九九久久| 久久影院资源站| 卡一卡二国产精品| 国产精品任我爽爆在线播放| 亚洲精品四区| 亚洲人成高清| 六月天综合网| 男女男精品网站| 99pao成人国产永久免费视频 | 日韩成人综合| 成人综合一区| 日韩电影免费网站| 亚洲成人精品| 91九色精品| 中文不卡在线| 欧美日韩91| 麻豆91在线播放| 国产一区二区三区久久久久久久久| 国产精品白丝一区二区三区| 成人亚洲一区| 国产精品99一区二区| 亚洲激情av| 中文字幕亚洲在线观看| 日韩中文字幕91| 日本麻豆一区二区三区视频| 欧美aa在线视频| 亚洲成人不卡| 日韩影院在线观看| 欧美在线日韩| 久久精品主播| 日产欧产美韩系列久久99| 麻豆视频一区二区| 欧美午夜精品一区二区三区电影| 蜜桃一区二区三区在线观看| 国产亚洲精品精品国产亚洲综合| 高清av不卡| 玖玖玖国产精品| 国产精品宾馆| 日韩在线视频精品| 日韩高清在线不卡| 免费视频亚洲| 国产精品久久久免费| 久久久久国产| 欧美国产另类| 韩日一区二区三区| 国产麻豆一区| av成人国产| 国产精品入口久久| 美女精品网站| 国精品一区二区| 久久精品国产久精国产| 国产精品日本| 波多视频一区| 在线视频亚洲欧美中文| 国产成人免费视频网站视频社区| 日韩在线a电影| 99成人超碰| av亚洲一区二区三区| 欧美精品99| 巨乳诱惑日韩免费av| 99国产精品99久久久久久粉嫩| 97在线精品| 国产66精品| 精品成av人一区二区三区 | 成人台湾亚洲精品一区二区| 国产精品毛片| 亚洲一级少妇| 少妇久久久久| 欧美xxxx中国| 精品福利久久久| 中文字幕人成乱码在线观看| 黄在线观看免费网站ktv| 欧美亚洲一区二区三区| 国产免费播放一区二区| 青青草91视频| 午夜亚洲福利| 欧美日韩一区自拍| 久久99免费视频| 国产精品久久久久77777丨| 久久久久久亚洲精品美女| 久久男人av资源站| 久久精品五月| 电影天堂国产精品| 国产欧美一区二区三区精品酒店| 高清不卡亚洲| 亚洲欧洲av| 最新亚洲一区| 欧美理论视频| 日韩天堂av| 男人的天堂亚洲一区| 久久午夜影视| 日韩综合一区二区三区| 麻豆国产一区| 99久久99久久精品国产片果冰| 日韩1区2区| 亚洲成a人片| 成人综合一区| 激情婷婷欧美| 亚洲激精日韩激精欧美精品| 日韩亚洲在线| 日韩影院在线观看| 国产精品a级| 久久精品播放| 日韩精品成人| 国产成人77亚洲精品www| 不卡在线一区| 国产精品久久久久久久久免费高清| 日本激情一区| 亚洲一二三区视频| 国产网站在线| 99国产精品私拍| 欧美在线观看天堂一区二区三区| 中文在线免费视频| 在线观看亚洲精品福利片| 欧美aa一级| 国产成人免费| 极品日韩av| 欧美中文字幕| 国产精品久久久久久久久久妞妞| 日韩大片在线| 蜜桃av一区二区| 天堂√中文最新版在线| aa亚洲婷婷| 欧美国产另类| 国产+成+人+亚洲欧洲在线| 美女黄网久久| 综合亚洲自拍| 亚洲www免费| 麻豆一区二区99久久久久| 99久精品视频在线观看视频| 亚洲免费黄色| 97久久超碰| 97精品中文字幕| av不卡在线看| 成人在线视频中文字幕| 伊人久久亚洲热| 精品久久97| 中文字幕中文字幕精品| 成人国产精品一区二区网站| 在线亚洲自拍| 成人羞羞在线观看网站| 欧美aa国产视频| 国产精品白丝一区二区三区| 欧美午夜不卡| 99精品视频精品精品视频| 欧美亚洲二区| 另类激情亚洲| 日韩电影免费在线观看| 精品网站aaa| 国产极品一区| 欧美一级二区|