欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解JavaScript中的閉包是如何產(chǎn)生的

 更新時間:2022年12月28日 10:43:04   作者:前端西瓜哥  
這篇文章主要為大家詳細(xì)介紹了從內(nèi)存管理的角度來看,JavaScript中的閉包是如何產(chǎn)生的。文中的示例代碼簡潔易懂,感興趣的小伙伴可以了解一下

這次從內(nèi)存管理的角度來看看,閉包是怎么產(chǎn)生的。

我們知道,在調(diào)用函數(shù)時,其實會產(chǎn)生臨時的 調(diào)用棧。這些調(diào)用棧保存的是 執(zhí)行上下本,并實際保存在 棧內(nèi)存 中。

每執(zhí)行一個函數(shù),函數(shù)內(nèi)的局部臨時變量會臨時保存起來。如果此時函數(shù)又調(diào)用了另一個函數(shù),另一個函數(shù)下的局部變量也要保存下來,就這樣,我們產(chǎn)生了棧。

當(dāng)一個函數(shù)執(zhí)行完后,它對應(yīng)的局部臨時變量就會被銷毀。

局部變量保存下來,是為了保護上下文現(xiàn)場。

舉例說明一下:

function a() {
  const a_num = 99;
  const a_obj = { val: "a" };
  b();
}

function b() {
  const b_str = "text";
  c();
}

function c() {
  const c_bool = true;
  // debugger
}

a();

這里我們嵌套調(diào)用了 a、b、c 函數(shù),會產(chǎn)生如下的調(diào)用棧。

基本類型的臨時變量,會直接保存到棧內(nèi)存中,對于引用類型,則是在堆內(nèi)存中生成,然后將地址拿到,保存到棧內(nèi)存中。

引用類型為什么不直接放到棧內(nèi)存中?因為棧內(nèi)存不是很大,很容易就棧溢出,而引用類型通常很大。

閉包的產(chǎn)生

函數(shù)調(diào)用完成后,它內(nèi)部聲明的臨時變量會被銷毀。理論上應(yīng)該如此,但如果使用了閉包,可以會讓臨時變量一直保留不被銷毀。

例子:

function createCounter() {
  let count = 0;
  let otherVal = "other val";
  return function counter() {
    // debugger;
    console.log(count++);
  };
}

const counter = createCounter();

console.log(counter());

執(zhí)行過程為:

  • 執(zhí)行函數(shù) createCounter 時,會創(chuàng)建一個空的上下文對象。
  • 遇到內(nèi)部函數(shù) counter,會 預(yù)掃描內(nèi)部函數(shù) counter 使用了 createCounter 下的哪些便利,最終掃描出 count 變量。于是在堆內(nèi)存創(chuàng)建一個閉包 Closure (createCounter) 對象,將 count 加進(jìn)去。otherVal 不會加到閉包對象上,因為它沒有被使用。
  • 這個內(nèi)部函數(shù)最后被返回,被引用,閉包就一直不會銷毀。

使用 DevTool 可以觀察到這個閉包對象:

所以,如果一個閉包返回的函數(shù)執(zhí)行完后不用了,要設(shè)置為 null。否則它關(guān)聯(lián)的閉包對象會一直在那里占用內(nèi)存。

多個內(nèi)部函數(shù)共享一個閉包對象

另外,如果有多個內(nèi)部函數(shù),這些函數(shù)會共用同一個閉包對象。即使其中的一個內(nèi)部函數(shù)不會返回,它也會給閉包對象加?xùn)|西。

下面我們加了一個 printOtherVal 的內(nèi)部函數(shù),它并不返回,但還是會導(dǎo)致返回 counter 函數(shù)對應(yīng)的閉包對象帶上了它不需要的 otherVal 變量。

這是 JS 引擎處理閉包策略問題,理論不應(yīng)該有這樣奇怪的效果。

結(jié)尾

調(diào)用函數(shù)時,會產(chǎn)生調(diào)用棧,將當(dāng)前函數(shù)上下文入棧,會保存基本類型變量。引用變量會在堆內(nèi)存中創(chuàng)建,然后在棧內(nèi)存中引用過來。

因為 JavaScript 中函數(shù)是第一公民,所以會有閉包的概念。當(dāng)發(fā)現(xiàn)內(nèi)部函數(shù),會創(chuàng)建一個閉包對象,將其中使用到的外部函數(shù)變量保存到該閉包對象下。之后內(nèi)部函數(shù)被調(diào)用時,就會從閉包里提取變量,如果找不到則從全局上下文提取。

到此這篇關(guān)于詳解JavaScript中的閉包是如何產(chǎn)生的的文章就介紹到這了,更多相關(guān)JavaScript閉包內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 原生js開發(fā)的日歷插件

    原生js開發(fā)的日歷插件

    本文主要分享了原生js開發(fā)的日歷插件的示例代碼。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-02-02
  • 淺談layui框架自帶分頁和表格重載的接口解析問題

    淺談layui框架自帶分頁和表格重載的接口解析問題

    今天小編就為大家分享一篇淺談layui框架自帶分頁和表格重載的接口解析問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-09-09
  • JS中常見的8種繼承方法總結(jié)

    JS中常見的8種繼承方法總結(jié)

    這篇文章主要給大家介紹了關(guān)于JS中常見的8種繼承方法,在js中所有的引用類型都繼承了Object,而繼承也是通過原型鏈實現(xiàn)的,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-10-10
  • JS控制div跳轉(zhuǎn)到指定的位置的幾種解決方案總結(jié)

    JS控制div跳轉(zhuǎn)到指定的位置的幾種解決方案總結(jié)

    這篇文章主要介紹了JS控制div跳轉(zhuǎn)到指定的位置的幾種解決方案總結(jié),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。
    2016-11-11
  • javascript入門之window對象【新手必看】

    javascript入門之window對象【新手必看】

    本文系統(tǒng)介紹了javascript的window對象以及一些控制函數(shù)的用法,僅供大家參考
    2016-11-11
  • js實現(xiàn)登陸遮罩效果的方法

    js實現(xiàn)登陸遮罩效果的方法

    這篇文章主要介紹了js實現(xiàn)登陸遮罩效果的方法,涉及javascript操作頁面元素樣式的相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-07-07
  • JavaScript實現(xiàn)九宮格移動拼圖游戲

    JavaScript實現(xiàn)九宮格移動拼圖游戲

    這篇文章主要為大家詳細(xì)介紹了JavaScript實現(xiàn)九宮格移動拼圖游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Bootstrap實現(xiàn)模態(tài)框效果

    Bootstrap實現(xiàn)模態(tài)框效果

    這篇文章主要為大家詳細(xì)介紹了Bootstrap實現(xiàn)模態(tài)框效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-09-09
  • JS調(diào)用打印方法設(shè)置頁眉頁腳的實例

    JS調(diào)用打印方法設(shè)置頁眉頁腳的實例

    一個網(wǎng)頁打印相關(guān)功能的擴展演示特效,在實現(xiàn)了打印功能外,還實現(xiàn)了打印預(yù)覽、打印前的頁眉頁腳設(shè)置,直接打印等功能,以前對JS打印前設(shè)置頁腳見的不多,所以這一個也算是挺有價值的,希望閑暇時參閱
    2013-05-05
  • xml文檔轉(zhuǎn)換工具,附圖表例子(hta)

    xml文檔轉(zhuǎn)換工具,附圖表例子(hta)

    xml文檔轉(zhuǎn)換程序的一個小應(yīng)用,幾行代碼生成餅圖。源代碼打包提供。
    2010-11-11

最新評論