成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

前端進(jìn)擊的巨人(一):執(zhí)行上下文與執(zhí)行棧,變量對象

_Suqin / 1858人閱讀

摘要:在中,通過棧的存取方式來管理執(zhí)行上下文,我們可稱其為執(zhí)行棧,或函數(shù)調(diào)用棧。而處于棧頂?shù)氖钱?dāng)前正在執(zhí)行函數(shù)的執(zhí)行上下文,當(dāng)函數(shù)調(diào)用完成后,它就會(huì)從棧頂被推出理想的情況下,閉包會(huì)阻止該操作,閉包后續(xù)文章深入詳解。

寫在開篇

已經(jīng)不敢自稱前端小白,曾經(jīng)吹過的牛逼總要一點(diǎn)點(diǎn)去實(shí)現(xiàn)。

正如前領(lǐng)導(dǎo)說的,自己喝酒吹過的牛皮,跪著都得含著淚去實(shí)現(xiàn)。

那么沒有年終完美總結(jié),來個(gè)新年莽撞開始可好。

進(jìn)擊巨人系列開篇,不忘初心,砥礪前行。

理解執(zhí)行上下文
執(zhí)行上下文(Execution Context): 函數(shù)執(zhí)行前進(jìn)行的準(zhǔn)備工作(也稱執(zhí)行上下文環(huán)境)

運(yùn)行JavaScript代碼時(shí),當(dāng)代碼執(zhí)行進(jìn)入一個(gè)環(huán)境時(shí),就會(huì)為該環(huán)境創(chuàng)建一個(gè)執(zhí)行上下文,它會(huì)在你運(yùn)行代碼前做一些準(zhǔn)備工作,如確定作用域,創(chuàng)建局部變量對象等。

具體做了什么先按下不表,先來看下JavaScript執(zhí)行環(huán)境有哪些?

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

全局環(huán)境

函數(shù)環(huán)境

eval函數(shù)環(huán)境 (已不推薦使用)

那么與之對應(yīng)的執(zhí)行上下文類型同樣有3種:

執(zhí)行上下文的類型

全局執(zhí)行上下文

函數(shù)執(zhí)行上下文

eval函數(shù)執(zhí)行上下文

JavaScript運(yùn)行時(shí)首先會(huì)進(jìn)入全局環(huán)境,對應(yīng)會(huì)生成全局上下文。程序代碼中基本都會(huì)存在函數(shù),那么調(diào)用函數(shù),就會(huì)進(jìn)入函數(shù)執(zhí)行環(huán)境,對應(yīng)就會(huì)生成該函數(shù)的執(zhí)行上下文。

先插播一個(gè)知識點(diǎn):"JS是單線程"! "單線程"! "單線程"!

簡單理解下單線程,就是同個(gè)時(shí)間段只能做一件任務(wù),完成之后才可以繼續(xù)下一個(gè)任務(wù)。正如女朋友只有一個(gè),各位面向?qū)ο蟮男』锇閭兡銈冋f對不對?有女票的必須說沒毛病。

既然是這樣,必須要有一個(gè)排隊(duì)機(jī)制,不然就會(huì)出現(xiàn)幾個(gè)流氓霸著車道不讓過,"還有王法么?"

JS中管理多個(gè)執(zhí)行上下文

函數(shù)編程中,代碼中會(huì)聲明多個(gè)函數(shù),對應(yīng)的執(zhí)行上下文也會(huì)存在多個(gè)。在JavaScript中,通過棧的存取方式來管理執(zhí)行上下文,我們可稱其為執(zhí)行棧,或函數(shù)調(diào)用棧(Call Stack)。

在說明執(zhí)行棧前,先來補(bǔ)下"棧數(shù)據(jù)結(jié)構(gòu)"知識點(diǎn)。

棧數(shù)據(jù)結(jié)構(gòu)


借助前端大神的例子,用乒乓球盒子來理解棧的存取方式。(這個(gè)例子讓我徹底記住了棧數(shù)據(jù)結(jié)構(gòu))

棧遵循"先進(jìn)后出,后進(jìn)先出"的規(guī)則,或稱LIFO ("Last In First Out") 規(guī)則。

如圖所示,我們只能從棧頂取出或放入乒乓球,最先放進(jìn)盒子的總是最后才能取出。
棧中"放入/取出",也可稱為"入棧/出棧"

總結(jié)棧數(shù)據(jù)結(jié)構(gòu)的特點(diǎn):

后進(jìn)先出,先進(jìn)后出

出口在頂部,且僅有一個(gè)

執(zhí)行棧(函數(shù)調(diào)用棧)

理解完棧的存取方式,我們接著分析JavaScript中如何通過棧來管理多個(gè)執(zhí)行上下文。

程序執(zhí)行進(jìn)入一個(gè)執(zhí)行環(huán)境時(shí),它的執(zhí)行上下文就會(huì)被創(chuàng)建,并被推入執(zhí)行棧中(入棧);
程序執(zhí)行完成時(shí),它的執(zhí)行上下文就會(huì)被銷毀,并從棧頂被推出(出棧),控制權(quán)交由下一個(gè)執(zhí)行上下文。

因?yàn)镴S執(zhí)行中最先進(jìn)入全局環(huán)境,所以處于"棧底的永遠(yuǎn)是全局環(huán)境的執(zhí)行上下文"。而處于"棧頂?shù)氖钱?dāng)前正在執(zhí)行函數(shù)的執(zhí)行上下文",當(dāng)函數(shù)調(diào)用完成后,它就會(huì)從棧頂被推出(理想的情況下,閉包會(huì)阻止該操作,閉包后續(xù)文章深入詳解)。

"全局環(huán)境只有一個(gè),對應(yīng)的全局執(zhí)行上下文也只有一個(gè),只有當(dāng)頁面被關(guān)閉之后它才會(huì)從執(zhí)行棧中被推出,否則一直存在于棧底"

文字太多不如上代碼系列 ——》代碼 + 圖,一覽無遺:

function foo () {
    function bar () {
        return "I am bar";
    }
    return bar();
}
foo();

執(zhí)行上下文的生命周期

執(zhí)行上下文的生命周期有兩個(gè)階段:

創(chuàng)建階段(進(jìn)入執(zhí)行上下文)

執(zhí)行階段(代碼執(zhí)行

創(chuàng)建階段:函數(shù)被調(diào)用時(shí),進(jìn)入函數(shù)環(huán)境,為其創(chuàng)建一個(gè)執(zhí)行上下文,此時(shí)進(jìn)入創(chuàng)建階段

執(zhí)行階段:執(zhí)行函數(shù)中代碼時(shí),此時(shí)執(zhí)行上下文進(jìn)入執(zhí)行階段

創(chuàng)建階段的操作

創(chuàng)建變量對象

函數(shù)環(huán)境會(huì)初始化創(chuàng)建Arguments對象(并賦值

函數(shù)聲明(并賦值

變量聲明,函數(shù)表達(dá)式聲明(未賦值

確定this指向(this由調(diào)用者確定

確定作用域(詞法環(huán)境決定,哪里聲明定義,就在哪里確定

執(zhí)行階段的操作

變量對象賦值

變量賦值

函數(shù)表達(dá)式賦值

調(diào)用函數(shù)

順序執(zhí)行其它代碼

看到這里,我們不經(jīng)會(huì)問變量對象是什么鬼,它與代碼中常見的函數(shù)聲明,變量聲明有神馬關(guān)系???

變量對象和活動(dòng)對象的區(qū)別:

當(dāng)進(jìn)入到一個(gè)執(zhí)行上下文后,這個(gè)變量對象才會(huì)被激活,所以叫活動(dòng)對象(AO),這時(shí)候活動(dòng)對象上的各種屬性才能被訪問。

"創(chuàng)建階段對函數(shù)聲明做賦值,變量及函數(shù)表達(dá)式僅做聲明,真正的賦值操作要等到執(zhí)行上下文代碼執(zhí)行階段"。

代碼例子1:變量提升

function foo() {
  console.log(a);         // 輸出undefined
  var a = "I am here";    // 賦值
}
foo();

// 實(shí)際執(zhí)行過程
function foo() {
  var a;                // 變量聲明,var初始化undefined
  console.log(a); 
  a = "I am here";     // 變量重新賦值
}

代碼例子2:函數(shù)聲明優(yōu)先級

function foo() {
    console.log(bar);
    var bar = 20;
    function bar() {
      return 10;
    }
    var bar = function() {
        return 30;
    }
}
foo();  // 輸出bar()整個(gè)函數(shù)聲明
函數(shù)聲明,變量聲明,函數(shù)表達(dá)式的優(yōu)先級

函數(shù)聲明,如果有同名屬性,會(huì)替換掉

變量,函數(shù)表達(dá)式

函數(shù)聲明優(yōu)先 > 變量,函數(shù)表達(dá)式

執(zhí)行上下文的數(shù)量限制(堆棧溢出)

執(zhí)行上下文可存在多個(gè),雖然沒有明確的數(shù)量限制,但如果超出棧分配的空間,會(huì)造成堆棧溢出。常見于遞歸調(diào)用,沒有終止條件造成死循環(huán)的場景。

// 遞歸調(diào)用自身
function foo() {
  foo();
}
foo();

// 報(bào)錯(cuò): Uncaught RangeError: Maximum call stack size exceeded
文末總結(jié)

JavaScript是單線程

棧頂?shù)膱?zhí)行上下文處于執(zhí)行中,其它需要排隊(duì)

全局上下文只有一個(gè)處于棧底,頁面關(guān)閉時(shí)出棧

函數(shù)執(zhí)行上下文可存在多個(gè),但應(yīng)避免遞歸時(shí)堆棧溢出

函數(shù)調(diào)用時(shí)就會(huì)創(chuàng)建新的上下文,即使調(diào)用自身,也會(huì)創(chuàng)建不同的執(zhí)行上下文

參考文檔:

執(zhí)行上下文詳細(xì)圖解

理解JavaScript 中的執(zhí)行上下文和執(zhí)行棧

這一次,徹底弄懂 JavaScript 執(zhí)行機(jī)制

本文首發(fā)Github,期待Star!
https://github.com/ZengLingYong/blog

作者:以樂之名
本文原創(chuàng),有不當(dāng)?shù)牡胤綒g迎指出。轉(zhuǎn)載請指明出處。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/101009.html

相關(guān)文章

  • 前端進(jìn)擊巨人(二):、堆、隊(duì)列、內(nèi)存空間

    摘要:中有三種數(shù)據(jù)結(jié)構(gòu)棧堆隊(duì)列。前端進(jìn)擊的巨人一執(zhí)行上下文與執(zhí)行棧,變量對象中解釋執(zhí)行棧時(shí),舉了一個(gè)乒乓球盒子的例子,來演示棧的存取方式,這里再舉個(gè)栗子搭積木。對于基本類型,棧中存儲的就是它自身的值,所以新內(nèi)存空間存儲的也是一個(gè)值。 面試經(jīng)常遇到的深淺拷貝,事件輪詢,函數(shù)調(diào)用棧,閉包等容易出錯(cuò)的題目,究其原因,都是跟JavaScript基礎(chǔ)知識不牢固有關(guān),下層地基沒打好,上層就是豆腐渣工程,...

    edgardeng 評論0 收藏0
  • 前端進(jìn)擊巨人(三):從作用域走進(jìn)閉包

    摘要:進(jìn)擊的巨人第三篇,本篇就作用域作用域鏈閉包等知識點(diǎn),一一擊破。在此我們遵照的方式,暫且稱是閉包。所以,一名合格的前端,除了會(huì)用閉包,還要正確的解除閉包引用。 進(jìn)擊的巨人第三篇,本篇就作用域、作用域鏈、閉包等知識點(diǎn),一一擊破。 showImg(https://segmentfault.com/img/bVburWd?w=1280&h=854); 作用域 作用域:負(fù)責(zé)收集并維護(hù)由所有聲明的...

    Vicky 評論0 收藏0
  • 前端進(jìn)擊巨人(四):略知函數(shù)式編程

    摘要:自執(zhí)行函數(shù)閉包實(shí)現(xiàn)模塊化以樂之名程序員產(chǎn)品經(jīng)理對作用域,以及閉包知識還沒掌握的小伙伴,可回閱前端進(jìn)擊的巨人三從作用域走進(jìn)閉包。參考文檔利用閉包實(shí)現(xiàn)模塊化翻譯淺談中的高階函數(shù)系列更文請關(guān)注專欄前端進(jìn)擊的巨人,不斷更新中。。。 系列更文前三篇文章,圍繞了一個(gè)重要的知識點(diǎn):函數(shù)。函數(shù)調(diào)用棧、函數(shù)執(zhí)行上下文、函數(shù)作用域到閉包。可見不理解函數(shù)式編程,代碼都擼不好。 showImg(https:/...

    omgdog 評論0 收藏0
  • 前端進(jìn)擊巨人(六):知否知否,須知this

    摘要:有關(guān)函數(shù)柯里化的詳解,請回閱前端進(jìn)擊的巨人五學(xué)會(huì)函數(shù)柯里化。構(gòu)造函數(shù)中的通過操作符可以實(shí)現(xiàn)對函數(shù)的構(gòu)造調(diào)用。在了解構(gòu)造函數(shù)中的前,有必要先了解下實(shí)例化對象的過程。 showImg(https://segmentfault.com/img/bVburMp?w=800&h=600); 常見this的誤解 指向函數(shù)自身(源于this英文意思的誤解) 指向函數(shù)的詞法作用域(部分情況) th...

    Andrman 評論0 收藏0
  • 前端進(jìn)擊巨人(八):淺談函數(shù)防抖節(jié)流

    摘要:隆重請出主角防抖與節(jié)流。防抖與節(jié)流的異同相同都是防止某一時(shí)間段內(nèi),函數(shù)被頻繁調(diào)用執(zhí)行,通過時(shí)間頻率控制,減少回調(diào)函數(shù)執(zhí)行次數(shù),來實(shí)現(xiàn)相關(guān)性能優(yōu)化。參考文章分鐘理解的節(jié)流防抖及使用場景函數(shù)防抖和節(jié)流 showImg(https://segmentfault.com/img/bVburM8?w=800&h=600); 本篇課題,或許早已是爛大街的解讀文章。不過春招系列面試下來,不少伙伴們還...

    _Zhao 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<