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 也隨之被銷毀了,即函數內的局部變量的生命周期僅存在于函數的聲明周期內,函數被銷毀,函數內的變量也自動被銷毀。
但是使用閉包就相反,函數執行完,生命周期結束,但是通過閉包引用的外層作用域內的變量依然存在,并且將一直存在,直到執行閉包的的作用域被銷毀,這里的局部變量才會被銷毀(如果在全局環境下引用了閉包,則只有在全局環境被銷毀,比如程序結束、瀏覽器關閉等行為時才會銷毀閉包引用的作用域)。因此為了避免閉包造成的內存損耗,建議在使用閉包后手動銷毀。還是上面示例二的例子,稍作修改:
新聞熱點
疑難解答
圖片精選