學(xué)習(xí)JS閉包

大名鼎鼎的閉包(closure),一直聽說很難,這個概念似乎玄之又玄秒咨,讓新手們望而卻步。為了弄懂JS閉包牡昆,也查閱了很多資料,下面是我的學(xué)習(xí)筆記摊欠,目的是弄清楚兩個問題:
1丢烘、什么是閉包?
2些椒、閉包的作用是什么播瞳?

一、變量的作用域


要理解閉包免糕,首先必須理解JS特殊的變量作用域赢乓。

變量的作用域無非就是兩種:全局變量和局部變量忧侧。

JS語言的特殊之處,就在于函數(shù)內(nèi)部可以直接讀取全局變量牌芋。

var n=999;
function f1(){
    alert(n);
}
f1(); // 999

另一方面蚓炬,在函數(shù)外部自然無法讀取函數(shù)內(nèi)部的局部變量

function f1(){
    var n = 999;
}
alert(n);  //error

這里有一個地方需要注意姜贡,函數(shù)內(nèi)部聲明變量時候试吁,一定要使用var命令。如果不用的話楼咳,你實際上聲明了一個全局變量。

function f1(){
    n=999;
}
f1();
alert(n); // 999

二烛恤、如何從外部讀取局部變量母怜?


出于種種原因,我們有時候需要得到函數(shù)內(nèi)的局部變量缚柏。但是苹熏,前面已經(jīng)說過了,正常情況下币喧,這是辦不到的轨域,只有通過變通方法才能實現(xiàn)。

那就是在函數(shù)的內(nèi)部杀餐,再定義一個函數(shù)干发。

function f1(){
    var n = 999;
    function f2(){
        alert(n);
    }
}

在上面的代碼中,函數(shù)f2就被包括在函數(shù)f1內(nèi)部史翘,這時f1內(nèi)部的所有局部變量枉长,對f2都是可見的。但是反過來就不行琼讽,f2內(nèi)部的局部變量必峰,對f1就是不可見得。這就是JS語言特有的“鏈?zhǔn)阶饔糜颉苯Y(jié)構(gòu)钻蹬,子對象會一級一級地向上尋找所有父對象的變量吼蚁。所以,父對象的所有變量问欠,對子對象都是可見的肝匆,反之則不成立。

既然f2可以讀取f1中的局部變量溅潜,那么只要把f2作為返回值术唬,我們不就可以在f1外部讀取它的內(nèi)部變量了嗎!

function f1(){
    var n = 999;
    function f2(){
        alert(n);
    }
    return f2;
}
var result = f1();
result();  //999

三滚澜、閉包的概念


上一節(jié)代碼中的f2函數(shù)粗仓,就是閉包。

閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)。

由于在JS語言中借浊,只有函數(shù)內(nèi)部的子函數(shù)才能讀取局部變量塘淑,因此可以把閉包簡單理解成“定義在一個函數(shù)內(nèi)部的函數(shù)”。

所以蚂斤,在本質(zhì)上存捺,閉包就是將函數(shù)內(nèi)部和函數(shù)外部連接起來的一座橋梁。

四曙蒸、閉包的用途


閉包可以用在許多地方捌治。它的最大用處有兩個,一個是前面提到的可以讀取函數(shù)內(nèi)部的變量纽窟,另一個就是讓這些變量的值始終保持在內(nèi)存中肖油。

看代碼:

  function f1(){
    var n=999;
    nAdd=function(){n+=1}
    function f2(){
      alert(n);
    }
    return f2;
  }
  var result=f1();
  result(); // 999
  nAdd();
  result(); // 1000

在這段代碼中,result實際上就是閉包f2函數(shù)臂港。它一共運(yùn)行了兩次森枪,第一次的值是999,第二次的值是1000审孽。這證明了县袱,函數(shù)f1中的局部變量n一直保存在內(nèi)存中,并沒有在f1調(diào)用后被自動清除佑力。

為什么會這樣呢式散?原因就在于f1是f2的父函數(shù),而f2被賦給了一個全局變量搓萧,這導(dǎo)致f2始終在內(nèi)存中杂数,而f2的存在依賴于f1,因此f1也始終在內(nèi)存中瘸洛,不會在調(diào)用結(jié)束后揍移,被垃圾回收機(jī)制回收。

這段代碼中另一個值得注意的地方反肋,就是nAdd=function(){n+=1}這行那伐,首先在nAdd前面沒有使用var命令,因此nAdd是一個全局變量石蔗,而不是局部變量罕邀。其次,nAdd的值是一個匿名函數(shù)养距,放在了f1內(nèi)部诉探,它本身就是一個閉包。

五棍厌、使用閉包的注意點


(1)肾胯、由于閉包會使得函數(shù)中的變量都被保存在內(nèi)存中竖席,內(nèi)存消耗很大,所以不能濫用閉包敬肚,否則會造成網(wǎng)頁的性能問題毕荐,在IE中可能導(dǎo)致內(nèi)存泄露。解決方法是艳馒,在退出函數(shù)之前憎亚,將不使用的局部變量全部刪除。
(2)弄慰、閉包會在父函數(shù)外部第美,改變父函數(shù)內(nèi)部變量的值。所以陆爽,如果你把父函數(shù)當(dāng)作對象(object)使用斋日,把閉包當(dāng)作它的公用方法(Public Method),把內(nèi)部變量當(dāng)作它的私有屬性(private value)墓陈,這時一定要小心,不要隨便改變父函數(shù)內(nèi)部變量的值第献。

(完)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末贡必,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子庸毫,更是在濱河造成了極大的恐慌仔拟,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,451評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件飒赃,死亡現(xiàn)場離奇詭異利花,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)载佳,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評論 3 394
  • 文/潘曉璐 我一進(jìn)店門炒事,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人蔫慧,你說我怎么就攤上這事挠乳。” “怎么了姑躲?”我有些...
    開封第一講書人閱讀 164,782評論 0 354
  • 文/不壞的土叔 我叫張陵睡扬,是天一觀的道長。 經(jīng)常有香客問我黍析,道長卖怜,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,709評論 1 294
  • 正文 為了忘掉前任阐枣,我火速辦了婚禮马靠,結(jié)果婚禮上奄抽,老公的妹妹穿的比我還像新娘。我一直安慰自己虑粥,他們只是感情好如孝,可當(dāng)我...
    茶點故事閱讀 67,733評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著娩贷,像睡著了一般第晰。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上彬祖,一...
    開封第一講書人閱讀 51,578評論 1 305
  • 那天茁瘦,我揣著相機(jī)與錄音,去河邊找鬼储笑。 笑死甜熔,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的突倍。 我是一名探鬼主播腔稀,決...
    沈念sama閱讀 40,320評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼羽历!你這毒婦竟也來了焊虏?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,241評論 0 276
  • 序言:老撾萬榮一對情侶失蹤秕磷,失蹤者是張志新(化名)和其女友劉穎诵闭,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體澎嚣,經(jīng)...
    沈念sama閱讀 45,686評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡疏尿,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,878評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了易桃。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片褥琐。...
    茶點故事閱讀 39,992評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖颈抚,靈堂內(nèi)的尸體忽然破棺而出踩衩,到底是詐尸還是另有隱情,我是刑警寧澤贩汉,帶...
    沈念sama閱讀 35,715評論 5 346
  • 正文 年R本政府宣布驱富,位于F島的核電站,受9級特大地震影響匹舞,放射性物質(zhì)發(fā)生泄漏褐鸥。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,336評論 3 330
  • 文/蒙蒙 一赐稽、第九天 我趴在偏房一處隱蔽的房頂上張望叫榕。 院中可真熱鬧浑侥,春花似錦、人聲如沸晰绎。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,912評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽荞下。三九已至伶选,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間尖昏,已是汗流浹背仰税。 一陣腳步聲響...
    開封第一講書人閱讀 33,040評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留抽诉,地道東北人陨簇。 一個月前我還...
    沈念sama閱讀 48,173評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像迹淌,于是被迫代替她去往敵國和親河绽。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,947評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 閉包(closure)是Javascript語言的一個難點唉窃,也是它的特色葵姥,很多高級應(yīng)用都要依靠閉包實現(xiàn)。 一句携、變量...
    zock閱讀 1,075評論 2 6
  • 閉包(closure)是Javascript語言的一個難點,也是它的特色允乐,很多高級應(yīng)用都要依靠閉包實現(xiàn)矮嫉。 下面就是...
    魯uin閱讀 300評論 0 2
  • 閉包(closure)是Javascript語言的一個難點,也是它的特色牍疏,很多高級應(yīng)用都要依靠閉包實現(xiàn)蠢笋。 下面就是...
    云端漫記閱讀 237評論 0 0
  • 第2章 基本語法 2.1 概述 基本句法和變量 語句 JavaScript程序的執(zhí)行單位為行(line),也就是一...
    悟名先生閱讀 4,149評論 0 13
  • 命令式標(biāo)題 1鳞陨、喂昨寞,培訓(xùn)員工前,培訓(xùn)下自己講故事的能力先厦滤! 2援岩、簽訂勞動合同,一定要注意這些事項掏导! 3享怀、25歲前你...
    NANA0閱讀 6,013評論 0 0