詳解JavaScript閉包問(wèn)題
閉包是純函數(shù)式編程語(yǔ)言的傳統(tǒng)特性之一。通過(guò)將閉包視為核心語(yǔ)言構(gòu)件的組成部分,JavaScript語(yǔ)言展示了其與函數(shù)式編程語(yǔ)言的緊密聯(lián)系。由于能夠簡(jiǎn)化復(fù)雜的操作,閉包在主流JavaScript庫(kù)以及高水平產(chǎn)品代碼中日益流行起來(lái)。
一、變量的作用域
在介紹閉包之前,我們先理解JavaScript的變量作用域。變量的作用域分為兩種:全局變量和局部變量。
1、全局變量
var n = 999; //全局變量 function f1() { a = 100; //在這里a也是全局變量 alert(n); } console.log(a); //100
在這里,函數(shù)內(nèi)外部可以直接取到變量的值——全局變量
2、局部變量
//局部變量 function f2() { var b = 22; } console.log(b); //報(bào)錯(cuò)
在這里,函數(shù)外部無(wú)法直接取到函數(shù)內(nèi)部定義的值——局部變量
講到這里,當(dāng)我們想要從外部取到局部變量的值,這時(shí)候該怎么辦呢?
請(qǐng)接著往下看:
二、如何從外部獲取局部變量
接下來(lái)我們看一個(gè)例子:
var outer = 'Outer'; // 全局變量 var copy; function outerFn(){ // 全局函數(shù) var inner = 'Inner'; // 該變量只有函數(shù)作用域,無(wú)法從外部訪問(wèn) function innerFn(){ // outerFn()中的innerFn() // 全局上下文和外圍上下文都可以在這里使用, // 因此可以訪問(wèn)到outer和inner console.log(outer); console.log(inner); } copy=innerFn; // 保存innerFn()的引用 // 因?yàn)閏opy是在全局上下文中聲明的,所以在外部可以使用 } outerFn(); copy(); // 不能直接調(diào)用innerFn(),但是可以通過(guò)在全局作用域中聲明的變量來(lái)調(diào)用
來(lái)分析一下上面的例子。在innerFn()中可以訪問(wèn)變量outer,因?yàn)樗幱谌稚舷挛闹小?/span>
在執(zhí)行完outerFn()之后,執(zhí)行了innerFn(),這是通過(guò)將該函數(shù)的引用復(fù)制到一個(gè)全局變量
copy中來(lái)實(shí)現(xiàn)的。在利用變量copy調(diào)用函數(shù)innerFn()執(zhí)行時(shí),此刻已經(jīng)不在outerFn()的作
用域中了。因此下面的代碼不是應(yīng)該失敗嗎?
console.log(inner);
變量inner的值應(yīng)該是undefined吧?可是,上面代碼片段的輸出卻是:
“Outer”
“Inner”
這就是JavaScript的鏈?zhǔn)阶饔糜蚪Y(jié)構(gòu),子對(duì)象會(huì)一級(jí)一級(jí)的向上尋找所有父對(duì)象的變量。所以父對(duì)象的所有變量對(duì)子對(duì)象都是可見(jiàn)的,反之則不成立。
這樣我們就可以獲取到函數(shù)內(nèi)部的局部變量了。
三、閉包的概念
上面代碼塊中的copy()函數(shù)就是閉包。在我的理解,閉包就是能夠讀取到函數(shù)內(nèi)部變量的函數(shù)。
而在JavaScript中,可以通過(guò)函數(shù)內(nèi)部的子函數(shù)獲取到局部變量,因此可以把閉包理解為定義在函數(shù)內(nèi)部的函數(shù)。
可以把它理解為一個(gè)將函數(shù)內(nèi)部和外部連接起來(lái)的橋梁。
四、閉包的作用
在我看來(lái),閉包的作用主要體現(xiàn)在兩個(gè)方面:
1、可以讀取函數(shù)內(nèi)部的變量
這個(gè)作用在上個(gè)代碼塊已經(jīng)表現(xiàn)得很清楚。
2、可以將局部變量的值一直保存在內(nèi)存中
總所周知,局部變量只有當(dāng)使用的時(shí)候才會(huì)在內(nèi)存中開(kāi)辟出暫時(shí)的存儲(chǔ)空間,在函數(shù)運(yùn)行結(jié)束后會(huì)自動(dòng)釋放空間。而閉包的出現(xiàn)可以使得局部變量可以像全局變量一樣一致存儲(chǔ)在內(nèi)存中。
function c1() { var z = 9999; nAdd = function() { z += 1; } function c2() { console.log(z); } return c2; } var result = c1(); result(); //9999 nAdd(); result(); //10000
在上述代碼中,先執(zhí)行一次c1(),此時(shí)z=9999;再執(zhí)行一次nAdd(),使z+1;在執(zhí)行一次c1()輸出此時(shí)z的值,z=10000。說(shuō)明z的值一直存儲(chǔ)在內(nèi)存中,并沒(méi)有在第一次調(diào)用c1()后背自動(dòng)消除。
此時(shí)就要注意,閉包的使用會(huì)消耗很大的內(nèi)存,不要濫用閉包。在退出函數(shù)之前,將不使用的局部變量全部刪除。
到此這篇關(guān)于詳解JavaScript閉包問(wèn)題的文章就介紹到這了,更多相關(guān)JavaScript閉包問(wèn)題內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Javascript入門(mén)學(xué)習(xí)第七篇 js dom實(shí)例操作
上篇文章我們講了 用dom方式 創(chuàng)建節(jié)點(diǎn),復(fù)制節(jié)點(diǎn),插入節(jié)點(diǎn)。 今天我們將講 刪除節(jié)點(diǎn),替換節(jié)點(diǎn),查找節(jié)點(diǎn)等。2008-07-07淺談javascript中onbeforeunload與onunload事件
javascript中onbeforeunload與onunload事件就是頁(yè)面加載前與頁(yè)面關(guān)閉時(shí)的兩個(gè)功能的函數(shù),可以防止頁(yè)面刷新時(shí)給提示再刷新或頁(yè)面關(guān)閉時(shí)給出提示,下面我來(lái)介紹onbeforeunload與onunload事件用法。2015-12-12javascript 打開(kāi)頁(yè)面window.location和window.open的區(qū)別
有時(shí)候需要用js來(lái)實(shí)現(xiàn)頁(yè)面的打開(kāi),因?yàn)閖s下有window.location和window.open的不同實(shí)現(xiàn)方法,下面來(lái)簡(jiǎn)單的說(shuō)明下區(qū)別。2010-03-03javascript正則匹配漢字、數(shù)字、字母、下劃線
javascript正則檢測(cè)只含有漢字、數(shù)字、字母、下劃線不能以下劃線開(kāi)頭和結(jié)尾,需要的朋友可以參考下2014-04-04jQuery入門(mén)問(wèn)答 整理的幾個(gè)常見(jiàn)的初學(xué)者問(wèn)題
大家可能看到了,我已經(jīng)將過(guò)去寫(xiě)的兩篇jQuery的教程刪掉了,因?yàn)楦郊淮嬖诹说鹊葐?wèn)題,所以刪除了,從今天開(kāi)始我就寫(xiě)jQuery教程第二版了!希望大家能夠支持我!2010-02-02javaScript面向?qū)ο罄^承方法經(jīng)典實(shí)現(xiàn)
很多人都說(shuō)JavaScript不能算是面向?qū)ο蟮淖兂烧Z(yǔ)言。但是JavaScript的類型非常松散,也沒(méi)有編譯器,但是我們可以模仿著其他語(yǔ)言實(shí)現(xiàn)面向?qū)ο蟮姆绞絹?lái)實(shí)現(xiàn)JavaScript的面向編程2013-08-08