詳解JavaScript閉包問題
閉包是純函數式編程語言的傳統特性之一。通過將閉包視為核心語言構件的組成部分,JavaScript語言展示瞭其與函數式編程語言的緊密聯系。由於能夠簡化復雜的操作,閉包在主流JavaScript庫以及高水平產品代碼中日益流行起來。
一、變量的作用域
在介紹閉包之前,我們先理解JavaScript的變量作用域。變量的作用域分為兩種:全局變量和局部變量。
1、全局變量
var n = 999; //全局變量 function f1() { a = 100; //在這裡a也是全局變量 alert(n); } console.log(a); //100
在這裡,函數內外部可以直接取到變量的值——全局變量
2、局部變量
//局部變量 function f2() { var b = 22; } console.log(b); //報錯
在這裡,函數外部無法直接取到函數內部定義的值——局部變量
講到這裡,當我們想要從外部取到局部變量的值,這時候該怎麼辦呢?
請接著往下看:
二、如何從外部獲取局部變量
接下來我們看一個例子:
var outer = 'Outer'; // 全局變量 var copy; function outerFn(){ // 全局函數 var inner = 'Inner'; // 該變量隻有函數作用域,無法從外部訪問 function innerFn(){ // outerFn()中的innerFn() // 全局上下文和外圍上下文都可以在這裡使用, // 因此可以訪問到outer和inner console.log(outer); console.log(inner); } copy=innerFn; // 保存innerFn()的引用 // 因為copy是在全局上下文中聲明的,所以在外部可以使用 } outerFn(); copy(); // 不能直接調用innerFn(),但是可以通過在全局作用域中聲明的變量來調用
來分析一下上面的例子。在innerFn()中可以訪問變量outer,因為它處於全局上下文中。
在執行完outerFn()之後,執行瞭innerFn(),這是通過將該函數的引用復制到一個全局變量
copy中來實現的。在利用變量copy調用函數innerFn()執行時,此刻已經不在outerFn()的作
用域中瞭。因此下面的代碼不是應該失敗嗎?
console.log(inner);
變量inner的值應該是undefined吧?可是,上面代碼片段的輸出卻是:
“Outer”
“Inner”
這就是JavaScript的鏈式作用域結構,子對象會一級一級的向上尋找所有父對象的變量。所以父對象的所有變量對子對象都是可見的,反之則不成立。
這樣我們就可以獲取到函數內部的局部變量瞭。
三、閉包的概念
上面代碼塊中的copy()函數就是閉包。在我的理解,閉包就是能夠讀取到函數內部變量的函數。
而在JavaScript中,可以通過函數內部的子函數獲取到局部變量,因此可以把閉包理解為定義在函數內部的函數。
可以把它理解為一個將函數內部和外部連接起來的橋梁。
四、閉包的作用
在我看來,閉包的作用主要體現在兩個方面:
1、可以讀取函數內部的變量
這個作用在上個代碼塊已經表現得很清楚。
2、可以將局部變量的值一直保存在內存中
總所周知,局部變量隻有當使用的時候才會在內存中開辟出暫時的存儲空間,在函數運行結束後會自動釋放空間。而閉包的出現可以使得局部變量可以像全局變量一樣一致存儲在內存中。
function c1() { var z = 9999; nAdd = function() { z += 1; } function c2() { console.log(z); } return c2; } var result = c1(); result(); //9999 nAdd(); result(); //10000
在上述代碼中,先執行一次c1(),此時z=9999;再執行一次nAdd(),使z+1;在執行一次c1()輸出此時z的值,z=10000。說明z的值一直存儲在內存中,並沒有在第一次調用c1()後背自動消除。
此時就要註意,閉包的使用會消耗很大的內存,不要濫用閉包。在退出函數之前,將不使用的局部變量全部刪除。
到此這篇關於詳解JavaScript閉包問題的文章就介紹到這瞭,更多相關JavaScript閉包問題內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!