国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 網站 > 幫助中心 > 正文

通俗易懂地解釋JS中的閉包

2024-07-09 22:42:07
字體:
來源:轉載
供稿:網友

1. "閉包就是跨作用域訪問變量。"

【示例一】

var name = 'wangxi'function user () { // var name = 'wangxi' function getName () { console.log(name) } getName()}user() // wangxi

在 getName 函數中獲取 name,首先在 getName 函數的作用域中查找 name,未找到,進而在 user 函數的作用域中查找,同樣未找到,繼續向上回溯,發現在全局作用域中存在 name,因此獲取 name 值并打印。這里很好理解,即變量都存在在指定的作用域中,如果在當前作用中找不到想要的變量,則通過作用域鏈向在父作用域中繼續查找,直到找到第一個同名的變量為止(或找不到,拋出 ReferenceError 錯誤)。這是 js 中作用域鏈的概念,即子作用域可以根據作用域鏈訪問父作用域中的變量,那如果相反呢,在父作用域想訪問子作用域中的變量呢?――這就需要通過閉包來實現。

【示例二】

function user () { var name = 'wangxi' return function getName () { return name }}var userName = user()()console.log(userName) // wangxi

分析代碼我們知道,name 是存在于 user 函數作用域內的局部變量,正常情況下,在外部作用域(這里是全局)中是無法訪問到 name 變量的,但是通過閉包(返回一個包含變量的函數,這里是 getName 函數),可以實現跨作用域訪問變量了(外部訪問內部)。因此上面的這種說法完整的應該理解為:

閉包就是跨作用域訪問變量 ―― 內部作用域可以保持對外部作用域中變量的引用從而使得(更)外部作用域可以訪問內部作用域中的變量。(還是不理解的話看下一條分析)

2. "閉包:在爺爺的環境中執行了爸爸,爸爸中返回了孫子,本來爸爸被執行完了,爸爸的環境應該被清除掉,但是孫子引用了爸爸的環境,導致爸爸釋放不了。這一坨就是閉包。簡單來講,閉包就是一個引用了父環境的對象,并且從父環境中返回到更高層的環境中的一個對象。"

這個怎么理解呢?首先看下方代碼:

【示例三】

function user () { var name = 'wangxi' return name}var userName = user()console.log(userName) // wangxi

問:這是閉包嗎?

答:當然不是。首先要明白閉包是什么。雖然這里形式上看好像也是在全局作用域下訪問了 user 函數內的局部變量 name,但是問題是,user 執行完,name 也隨之被銷毀了,即函數內的局部變量的生命周期僅存在于函數的聲明周期內,函數被銷毀,函數內的變量也自動被銷毀。

但是使用閉包就相反,函數執行完,生命周期結束,但是通過閉包引用的外層作用域內的變量依然存在,并且將一直存在,直到執行閉包的的作用域被銷毀,這里的局部變量才會被銷毀(如果在全局環境下引用了閉包,則只有在全局環境被銷毀,比如程序結束、瀏覽器關閉等行為時才會銷毀閉包引用的作用域)。因此為了避免閉包造成的內存損耗,建議在使用閉包后手動銷毀。還是上面示例二的例子,稍作修改:

【示例四】

function user () { var name = 'wangxi' return function getName () { return name }}var userName = user()() // userName 變量中始終保持著對 name 的引用console.log(userName) // wangxiuserName = null // 銷毀閉包,釋放內存

【為什么 user()() 是兩個括號:執行 user()  返回的是 getName 函數,要想獲得 name 變量,需要對返回的 getName 函數執行一次,所以是 user()()】

根據觀點2,分析一下代碼:在全局作用域下創建了 userName 變量(爺爺),保存了對 user 函數最終返回結果的引用(即局部變量 name 的值),執行 user()()(爸爸),返回了 name(孫子),正常情況下,在執行了 user()() 之后,user 的環境(爸爸)應該被清除掉,但是因為返回的結果 name(孫子)引用了爸爸的環境(因為 name 本來就是存在于 user 的作用域內的),導致 user 的環境無法被釋放(會造成內存損耗)。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 栾川县| 秦皇岛市| 赤壁市| 武胜县| 营口市| 台安县| 新建县| 裕民县| 遂溪县| 咸阳市| 洛阳市| 怀来县| 朔州市| 巴塘县| 夏邑县| 广南县| 离岛区| 靖州| 石嘴山市| 远安县| 台前县| 武宣县| 英吉沙县| 盖州市| 怀安县| 黑山县| 平湖市| 淳化县| 雅江县| 黄大仙区| 红原县| 洪湖市| 武胜县| 明光市| 乡宁县| 阳新县| 德安县| 宜黄县| 正安县| 界首市| 淄博市|