摘要:前言這段時間一直在消化作用域鏈和閉包的相關(guān)知識。而作用域鏈則是這套規(guī)則這套規(guī)則的具體運(yùn)行。是變量對象的縮寫那這樣放有什么好處呢我們知道作用域鏈保證了當(dāng)前執(zhí)行環(huán)境對符合訪問權(quán)限的變量和函數(shù)的有序訪問。
前言:這段時間一直在消化作用域鏈和閉包的相關(guān)知識。之前看《JS高程》和一些技術(shù)博客,對于這些概念的論述多多少少不太清楚或者不太完整,包括一些大神的技術(shù)文章。這也給我的學(xué)習(xí)上造成了一些困惑,這幾個概念的理解也是始終處于一個半懂不懂的狀態(tài)。后來在某公眾號看到了@波同學(xué)的基礎(chǔ)文章,這應(yīng)該是我所看到的最清楚,最全面,最好懂的文章了。所以我在學(xué)習(xí)之余決定寫一篇文章,總結(jié)學(xué)到的知識點,用我的理解來闡述,不足之處,見請諒解。執(zhí)行上下文(Execution Context)
也叫執(zhí)行環(huán)境,也可以簡稱“環(huán)境”。是JS在執(zhí)行過程中產(chǎn)生的,當(dāng)JS執(zhí)行一段可執(zhí)行的代碼時,就會生成一個叫執(zhí)行環(huán)境的東西。JS中每個函數(shù)都會有自己的執(zhí)行環(huán)境,當(dāng)函數(shù)執(zhí)行時,就生成了它的執(zhí)行環(huán)境,執(zhí)行上下文會生成函數(shù)的作用域。
除了函數(shù)有執(zhí)行環(huán)境,還有全局的環(huán)境。在JS中,往往不止一個執(zhí)行環(huán)境。
讓我們先來看一個栗子:
var a=10; function foo(){ var b=5; function fn(){ var c=20; var d=100; } fn(); } foo();
在這個栗子中,包括了三個執(zhí)行環(huán)境:全局環(huán)境,foo()執(zhí)行環(huán)境,fn()執(zhí)行環(huán)境;
執(zhí)行環(huán)境的處理機(jī)制在這里我們要了解到執(zhí)行上下文的第一個特點:內(nèi)部的環(huán)境可以訪問外部的環(huán)境,而外部的環(huán)境無法訪問內(nèi)部的環(huán)境。
例如:我們可以在fn()中訪問到位于foo()中的b,在全局環(huán)境中的a,而在foo()中卻無法訪問到c或者d。
為什么會這樣,這就要了解JS處理代碼的一個機(jī)制了。
我們知道JS的處理過程是以堆棧的方式來處理,JS引擎會把執(zhí)行環(huán)境一個個放入棧里,然后先放進(jìn)去的后處理,后放進(jìn)去的先處理,上面這個栗子,最先被放進(jìn)棧中的是全局環(huán)境,然后是foo(),再是fn(),然后處理完一個拿出一個來,所以我們知道為什么foo()不能訪問fn()里的了,因為它已經(jīng)走了。
執(zhí)行環(huán)境的生命周期好了,了解完執(zhí)行環(huán)境的的處理方式,我們要說明執(zhí)行環(huán)境的生命周期。
執(zhí)行環(huán)境的生命周期分為兩個階段,這兩個階段描述了執(zhí)行環(huán)境在棧里面做了些什么。
創(chuàng)建階段創(chuàng)建階段;
執(zhí)行階段
執(zhí)行環(huán)境在創(chuàng)建階段會完成這么幾個任務(wù):1.生成變量對象;2.建立作用域鏈;3.確定this指向
執(zhí)行階段到了執(zhí)行階段,會給變量賦值,函數(shù)引用,然后還有執(zhí)行其他的代碼。
完成了這兩個步驟,執(zhí)行環(huán)境就可以準(zhǔn)備出棧,一路走好了。
以上就是執(zhí)行環(huán)境的具體執(zhí)行內(nèi)容。上面提到了執(zhí)行環(huán)境在創(chuàng)建階段會生成變量對象,這也是一個很重要的概念,我們下文會詳細(xì)論述。
變量對象(variable object)變量對象是什么呢?《JS高程》是這樣說的:“每個執(zhí)行環(huán)境都有與之關(guān)聯(lián)的變量對象,環(huán)境中定義的所有變量和函數(shù)都保存在這個對象中。”
那變量對象里有些什么東西呢?看下文:
變量對象的內(nèi)容在變量對象創(chuàng)建時,經(jīng)過了這樣三個步驟:
生成arguments屬性;
找到function函數(shù)聲明,創(chuàng)建屬性;
找到var變量聲明,創(chuàng)建屬性
其中值得注意的是:function函數(shù)聲明的級別比var變量聲明的級別要高,所以在實際執(zhí)行的過程中會先尋找function的聲明。
還需要注意的是:在執(zhí)行環(huán)境的執(zhí)行階段之前,變量對象中的屬性都無法訪問,這里還有一個活動對象(activation object)的概念,其實這個概念正是由進(jìn)入執(zhí)行階段的變量對象轉(zhuǎn)化而來。
來看一個栗子:
function foo(){ var a=10; function fn(){ return 5; } } foo();
讓我們來看看foo()函數(shù)的執(zhí)行環(huán)境:
它會包括三個部分:1.變量對象;2.作用域鏈;3.this指向?qū)ο?/p> 創(chuàng)建階段:
建立arguments
找到fn();
找到變量a,undefined;
執(zhí)行階段:變量對象變成活動對象;
arguments還是它~
fn();
a=10;
以上就是變量對象的內(nèi)容了,需要記住這個東西,因為會方便我們了解下文另一個重要的概念:作用域鏈。
作用域鏈(scope chain)什么是作用域鏈?《JS高程》里的文字是:“作用域鏈的用途,是保證對執(zhí)行環(huán)境有權(quán)訪問的所有變量和函數(shù)的有序訪問?!便虏汇卤疲糠凑业谝淮慰吹降臅r候確實是懵逼了。前面我們說過作用域,那么作用域鏈?zhǔn)遣皇蔷褪谴谝黄鸬淖饔糜蚰??并不是?/pre>作用域和作用域鏈的關(guān)系,用@波同學(xué)的話說,作用域是一套通過標(biāo)識符查找變量的規(guī)則。而作用域鏈則是這套規(guī)則這套規(guī)則的具體運(yùn)行。
是不是還是有點懵逼?還是看例子吧:
function foo(){ var a=10; function fn(){ return 5; } } foo();我們還是用上面的栗子,這次我們只看作用域鏈,根據(jù)規(guī)則,在一個函數(shù)的執(zhí)行環(huán)境的作用域鏈上,會依次放入自己的變量對象,父級的變量對象,祖級的變量對象.....一直到全局的變量對象。
比如上面這個栗子,fn()的執(zhí)行環(huán)境的作用域鏈上會有些什么呢?首先是自己的OV,然后是foo()的OV,接著就是全局的OV。而foo()的作用域鏈則會少一個fn()的OV。(OV是變量對象的縮寫)
那這樣放有什么好處呢?我們知道“作用域鏈保證了當(dāng)前執(zhí)行環(huán)境對符合訪問權(quán)限的變量和函數(shù)的有序訪問?!庇行颍⊥鈱雍瘮?shù)不能訪問內(nèi)層函數(shù)的變量,而內(nèi)層能夠訪問外層。正是有了這個作用域鏈,通過這個有方向的鏈,我們可以查找標(biāo)識符,進(jìn)而找到變量,才能實現(xiàn)這個特性。
閉包好了,終于要講到這個前端小萌新眼里的小boss了。在技術(shù)博客和書里翻滾了將將一周,對閉包的各種解釋把我搞得精力憔悴,懷疑人生。以至于在寫下這段關(guān)于閉包的論述時,也是內(nèi)心忐忑,因為我也不確定我說的是百分之百正確。先看看《JS高程》說的:“閉包是指有權(quán)訪問另一個函數(shù)作用域中的變量的函數(shù)。”
@波同學(xué)的說法是:“當(dāng)函數(shù)可以記住并訪問所在的作用域(全局作用域除外)時,就產(chǎn)生了閉包,即使函數(shù)是在當(dāng)前作用域之外執(zhí)行?!?/p>
......
好吧其實我覺得都說的不是太清楚。讓我們這樣來理解,就是內(nèi)部函數(shù)引用了外部函數(shù)的變量對象時,外部函數(shù)就是一個閉包。
還是看例子吧。
function foo(){ var a=20; return function(){ return a; } } foo()();在這個栗子中,foo()函數(shù)內(nèi)部返回了一個匿名函數(shù),而匿名函數(shù)內(nèi)部引用了外部函數(shù)foo()的變量a,由于作用域鏈,這個引用是有效的,按照J(rèn)S的機(jī)制,foo()執(zhí)行完畢后,執(zhí)行環(huán)境會失去引用,內(nèi)存會銷毀,但是由于內(nèi)部的匿名函數(shù)的引用,a會被暫時保存下來,罩著a的就是閉包。
return一個匿名函數(shù)時創(chuàng)造一個閉包的最簡單的方式,實際上創(chuàng)造閉包十分靈活,再看一個栗子:
var fn = null; function foo() { var a = 2; function innnerFoo() { console.log(a); } fn = innnerFoo; } function bar() { fn(); } foo(); bar(); // 2栗子來自@波同學(xué);
如上,可以看到:通過把innnerFoo()賦值給全局變量fn,內(nèi)部的函數(shù)在當(dāng)前作用域外執(zhí)行了,但是這不會影響foo形成了一個閉包。
閉包和兩個不同的案例這兩組栗子都是在各種書籍和各種博客上司空見慣了的栗子,其實跟閉包的關(guān)系不是很大,但是涉及到了函數(shù)相關(guān)的知識點,所以在這里寫下來。也算是積累。閉包和變量(見《JS高程》P181)一個例子
function createFunction(){ var result=new Array(); for(i=0;i<10;i++){ result[i]=function(){ return i; } } return result; } alert(createFunction());這個例子并不會如我們以為的返回從0到9的一串索引值。
當(dāng)我們執(zhí)行createFunction()時,函數(shù)內(nèi)會return result,而我們注意到result是一個數(shù)組,而每一個result[i]呢?它返回的則是一個函數(shù),而不是這個函數(shù)的執(zhí)行結(jié)果 i。所以我們想要返回一串索引值的時候,試著選擇result數(shù)組的其中一個,再加上圓括號讓它執(zhí)行起來,像這樣:
createFunction()[2]()這樣子就能執(zhí)行了嗎?運(yùn)行起來發(fā)現(xiàn)并沒有,執(zhí)行的結(jié)果是一串的i,為什么呢?
原因是在執(zhí)行createFunction()的時候,i的值已經(jīng)增加到了10,即退出循環(huán)的值,而再要執(zhí)行result內(nèi)部的匿名函數(shù)時,它能獲取到的i就只有10了,所以不管引用多少次,i的值都會是10;
那要如何修改才能達(dá)到我們的目的呢?
function createFunction(){ var result=[]; for(i=0;i<10;i++){ result[i]=function(num){ return function(){ return num; }; }(i); } return result; } alert(createFunction()[2]());彈出的警告和索引值一模一樣。這又是什么原因呢?
我們執(zhí)行createFunction()時,把外部的匿名函數(shù)的執(zhí)行結(jié)果賦值給了result,返回的result就是十個函數(shù)的數(shù)組。
而在這個外部函數(shù)里,有一個參數(shù)num,由于IIFE(立即執(zhí)行函數(shù))的緣故,循環(huán)過程中的i被賦值給了一個個的num,前后一共保存了10個num,為什么能夠保存下來呢?因為內(nèi)部的匿名函數(shù)引用了num。而這外部函數(shù)就是一個閉包
接下來,當(dāng)執(zhí)行createFunction()[2]()時實際上是執(zhí)行這個數(shù)組result的第三項,即:
function(){ return num; };這個函數(shù)。
num值是多少呢?如前所述,正是對應(yīng)的i。所以返回的值就能夠達(dá)到我們的預(yù)期了。
實際上,我認(rèn)為這個例子中更重要的是自執(zhí)行函數(shù)這個概念,正是有了自執(zhí)行,才能形成多對對多的引用,盡管這個例子里確實存在閉包,不過我認(rèn)為用這個例子來介紹閉包并不是太恰當(dāng)。閉包和thisthis也是JS里一個重中之重。我們知道,JS的this十分靈活的,前面已經(jīng)介紹過,this的指向在函數(shù)執(zhí)行環(huán)境建立時確定。函數(shù)中的this的指向是一個萌新們的難點,什么時候它是指向全局環(huán)境呢?什么時候它又是指向?qū)ο竽??注意:此處討論的是指函?shù)中的this,全局環(huán)境下的this一般情況指向window。結(jié)論一:this的指向是在函數(shù)被調(diào)用的時候確定的因為當(dāng)一個函數(shù)調(diào)用時,一個執(zhí)行環(huán)境就創(chuàng)建了,接著它會執(zhí)行,這是執(zhí)行環(huán)境的生命周期。所以this的指向是在函數(shù)被調(diào)用時確定的。
結(jié)論二:當(dāng)函數(shù)執(zhí)行時,如果這個函數(shù)是屬于某個對象,調(diào)用的方式是以對象的方法進(jìn)行的,那么this的指向就是這個對象,而其他情況,如函數(shù)獨立調(diào)用,則基本是指向全局對象。PS:實際上這個說法不大準(zhǔn)確,當(dāng)函數(shù)獨立調(diào)用時,在嚴(yán)格模式下,this的指向時undefined,而非嚴(yán)格模式下,則時指向全局對象。
為了更好的說明,讓我們看一個例子:
var a = 20; var foo = { a: 10, getA: function () { return this.a; } } console.log(foo.getA()); // 10 var test = foo.getA; console.log(test()); // 20在上面這個例子中,foo.getA()作為對象方法的調(diào)用,指向的自然是這個對象,而test雖然指向和foo.getA相同,但是因為是獨立調(diào)用,所以在非嚴(yán)格模式下,指向的是全局對象。
除了上面的例子,在《JS高程》中還有一個經(jīng)典的例子,眾多博客文章均有討論,但是看過之后覺得解釋還是不夠清楚,至少我沒完全理解,這里我將試著用自己的語言來解釋。var name="the window"; var object={ name:"my object", getNameFunc:function(){ return function(){ return this.name; }; } }; alert(object.getNameFunc()()); // the window在這個帶有閉包的例子里,我們可以看到object.getNameFunc()執(zhí)行的返回是一個函數(shù),再加()執(zhí)行則是一個直接調(diào)用了。所以指向的是全局對象。
如果我們想要返回變量對象怎么辦呢?
讓我們看一段代碼:
var name="the window";
var object={ name:"my object", getFunc:function(){ return this.name; } }; alert(object.getFunc()); //"my object"```我去掉了上面例子的閉包,可以看出在方法調(diào)用的情況下,this指向的是對象,那么我們只要在閉包能訪問到的位置,同時也是在這個方法調(diào)用的同一個作用域里設(shè)置一個“中轉(zhuǎn)站”就好了,讓我們把這個位置的this賦值給一個變量來存儲,然后匿名函數(shù)調(diào)用這個變量時指向的就會是對象而不是全局對象了。
var name="the window"; var object={ name:"my object", getFunc:function(){ var that=this; return function(){ return that; }; } }; alert(object.getFunc());that"s all
閉包的應(yīng)用閉包的應(yīng)用太多了,最重要的一個就是模塊模式了。不過說實話,實在還沒上路,所以這里就用一個模塊的栗子來結(jié)尾吧。(強(qiáng)行結(jié)尾)(function () { var a = 10; var b = 20; function add(num1, num2) { var num1 = !!num1 ? num1 : a; var num2 = !!num2 ? num2 : b; return num1 + num2; } window.add = add; })(); add(10, 20);我們需要知道的是,所謂模塊利用的就是閉包外部無法訪問內(nèi)部,內(nèi)部卻能訪問外部的特性,通過引用了指定的公共變量和方法,達(dá)到訪問私有變量和方法的目的。模塊可以保證模塊內(nèi)部的私有方法和變量不被外部變量污染,進(jìn)而方便更大規(guī)模的開發(fā)項目。
so,這篇文就到這里辣,寫了一個下午,最最最要感謝的是@波同學(xué),正是讀了他出色的教程,才能讓我對JS的理解更深一點,他的每一篇技術(shù)文章都是非常用心的,事實上,我覺得我的論述仍然不夠系統(tǒng)清晰,想要了解得更清晰的朋友可以去簡書搜索@波同學(xué)閱讀他寫得技術(shù)文章,好了,就這樣,債見
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/107068.html
摘要:在此例中,在匿名函數(shù)被返回后,它的作用域鏈初始化為包含函數(shù)的活動對象和全局變量對象。函數(shù)在執(zhí)行完畢后,其活動對象也不會被銷毀,因為匿名函數(shù)的作用域鏈仍然在引用這個活動對象,結(jié)果就是只是的執(zhí)行環(huán)境的作用域鏈會被銷毀,其活動對象會留在內(nèi)存中。 寫在前面 注:這個系列是本人對js知識的一些梳理,其中不少內(nèi)容來自書籍:Javascript高級程序設(shè)計第三版和JavaScript權(quán)威指南第六版,...
摘要:本期推薦文章從作用域鏈談閉包,由于微信不能訪問外鏈,點擊閱讀原文就可以啦。推薦理由這是一篇譯文,深入淺出圖解作用域鏈,一步步深入介紹閉包。作用域鏈的頂端是全局對象,在全局環(huán)境中定義的變量就會綁定到全局對象中。 (關(guān)注福利,關(guān)注本公眾號回復(fù)[資料]領(lǐng)取優(yōu)質(zhì)前端視頻,包括Vue、React、Node源碼和實戰(zhàn)、面試指導(dǎo)) 本周開始前端進(jìn)階的第二期,本周的主題是作用域閉包,今天是第6天。 本...
摘要:變量對象也是有父作用域的。作用域鏈的頂端是全局對象。當(dāng)函數(shù)被調(diào)用的時候,作用域鏈就會包含多個作用域?qū)ο?。?dāng)函數(shù)要訪問時,沒有找到,于是沿著作用域鏈向上查找,在的作用域找到了對應(yīng)的標(biāo)示符,就會修改的值。 一、概要 對于閉包的定義(紅寶書P178):閉包就是指有權(quán)訪問另外一個函數(shù)的作用域中的變量的函數(shù)。 關(guān)鍵點: 1、閉包是一個函數(shù) 2、能夠訪問另外一個函數(shù)作用域中的變量 二、閉包特性 對...
摘要:所以,當(dāng)在函數(shù)中使用全局變量的時候,所產(chǎn)生的代價是最大的,因為全局對象一直處于作用域鏈的最末位置,讀取局部變量是最快的。 什么是作用域 在編程語言中,作用域控制著變量與參數(shù)的可見性及生命周期,它能減少名稱沖突,而且提供了自動內(nèi)存管理(javascript 語言精粹) 靜態(tài)作用域 再者,js不像其他的編程語言一樣,擁有著塊級作用域,就像下面一段代碼。 function afunction...
摘要:為了防止之后自己又開始模糊,所以自己來總結(jié)一下中關(guān)于作用域鏈和原型鏈的知識,并將二者相比較看待進(jìn)一步加深理解。因此我們發(fā)現(xiàn)當(dāng)多個作用域相互嵌套的時候,就形成了作用域鏈。原型鏈原型說完了作用域鏈,我們來講講原型鏈。 畢業(yè)也整整一年了,看著很多學(xué)弟都畢業(yè)了,忽然心中頗有感慨,時間一去不復(fù)還呀。記得從去年這個時候接觸到JavaScript,從一開始就很喜歡這門語言,當(dāng)時迷迷糊糊看完了《J...
閱讀 2013·2021-09-22 16:05
閱讀 9335·2021-09-22 15:03
閱讀 2893·2019-08-30 15:53
閱讀 1706·2019-08-29 11:15
閱讀 914·2019-08-26 13:52
閱讀 2360·2019-08-26 11:32
閱讀 1809·2019-08-26 10:38
閱讀 2575·2019-08-23 17:19