什么是閉包?有何作用硝桩?

大家好沿猜,我是IT修真院的學(xué)員碗脊,一枚正直純潔善良的web前端程序員。

今天給大家?guī)淼氖情]包是什么衙伶,用處如何?

1.背景介紹

閉包(closure)是JS中一個(gè)較難理解的一個(gè)概念矢劲,JS函數(shù)的執(zhí)行依賴于變量作用域,函數(shù)對象的內(nèi)部狀態(tài)包含函數(shù)自身的邏輯另绩,還必須引用當(dāng)前的作用域鏈。函數(shù)對象可以相互關(guān)聯(lián)起來花嘶,函數(shù)體內(nèi)部的變量可以保存在函數(shù)作用域內(nèi),具有這種特性的函數(shù)稱為閉包车海。從這個(gè)概念上講,所有的函數(shù)都是閉包侍芝。

2.知識剖析

閉包可以用在許多地方埋同。它的最大用處有兩個(gè):

1.可以讀取函數(shù)內(nèi)部的變量

2.讓這些變量的值始終保存在內(nèi)存中

讀取函數(shù)內(nèi)部的變量的例子:

/*使用閉包讀取函數(shù)內(nèi)部的變量*/

function f1(){

n = 999;

function f2(){

alert(n);

}

return f2;

}

var result = f1();

result(); //999

在上面的代碼中,函數(shù)f2就被包括在函數(shù)f1內(nèi)部凶赁,這時(shí)f1內(nèi)部的所有局部變量逆甜,對f2都是可見的。但是反過來就不行交煞,f2內(nèi)部的局部變量斟或,對f1就是不可見的素征。這就是Javascript語言特有的"鏈?zhǔn)阶饔糜?結(jié)構(gòu)(chain scope)萝挤,子對象會一級一級地向上尋找所有父對象的變量。所以怜珍,父對象的所有變量,對子對象都是可見的绘面,反之則不成立侈沪。既然f2可以讀取f1中的局部變量,那么只要把f2作為返回值亭罪,我們不就可以在f1外部讀取它的內(nèi)部變量了嗎?

變量的值始終保存在內(nèi)存的例子:

/*使用閉包讓函數(shù)內(nèi)部的變量儲存在內(nèi)存中*/

function f1(){

n = 999;

nAdd = function(){

n+=1;

};

function f2(){

alert(n);

}

return f2;

}

var result = f1();

result();//999

nAdd();

result();//1000

在這段代碼中,result實(shí)際上就是閉包f2函數(shù)情组。它一共運(yùn)行了兩次,第一次的值是999箩祥,第二次的值是1000。這證明了袍祖,函數(shù)f1中的局部變量n一直保存在內(nèi)存中,并沒有在f1調(diào)用后被自動(dòng)清除捐凭。因?yàn)閒1是f2的父函數(shù),而f2被賦給了一個(gè)全局變量茁肠,這導(dǎo)致f2始終在內(nèi)存中缩举,而f2的存在依賴于f1垦梆,因此f1也始終在內(nèi)存中匹颤,不會在調(diào)用結(jié)束后,被垃圾回收機(jī)制(garbage collection)回收惋嚎。這段代碼中另一個(gè)值得注意的地方站刑,就是"nAdd=function(){n+=1}"這一行,首先在nAdd前面沒有使用var關(guān)鍵字绞旅,因此nAdd是一個(gè)全局變量,而不是局部變量因悲。其次,nAdd的值是一個(gè)匿名函數(shù)(anonymous function)晃琳,而這個(gè)匿名函數(shù)本身也是一個(gè)閉包讯检,所以nAdd相當(dāng)于是一個(gè)setter,可以在函數(shù)外部對函數(shù)內(nèi)部的局部變量進(jìn)行操作人灼。

3.常見問題

window.onload = function(){

var el = document.getElementById("id");

el.onclick = function(){

alert(el.id);

}

}

這段代碼會造成內(nèi)存泄漏顾翼,為什么投放?

4.解決方案

內(nèi)存泄漏的原因:執(zhí)行這段代碼的時(shí)候适贸,將匿名函數(shù)對象賦值給el的onclick屬性;然后匿名函數(shù)內(nèi)部又引用了el對象拜姿,存在循環(huán)引用,所以不能被垃圾回收機(jī)制回收误阻;

修改后:

window.onload = function(){

var el = document.getElementById("id");

var id = el.id; //解除循環(huán)引用

el.onclick = function(){

alert(id);

}

el = null; // 將閉包引用的外部函數(shù)中活動(dòng)對象清除

}

5.編碼實(shí)戰(zhàn)

在函數(shù)執(zhí)行過程中晴埂,為讀取和寫入變量的值,就需要在作用域鏈中查找變量

function compare(value1, value2){

if (value1 < value2){

return -1;

} else if (value1 > value2){

return 1;

} else {

return 0;

}

}

var result = compare(5, 10);

以上代碼先定義了compare()函數(shù)儒洛,然后又在全局作用域中調(diào)用了它。

6.擴(kuò)展思考

閉包有什么優(yōu)缺點(diǎn)琅锻,何時(shí)使用向胡?

7.參考文獻(xiàn)

參考一:阮一峰的網(wǎng)絡(luò)日志:學(xué)習(xí)Javascript閉包

參考二:知乎專欄:JS中的閉包是什么惊完?

參考三:segmentfault:JS進(jìn)階之閉包

8.更多討論

討論一:在閉包中的this指向問題?

this是一個(gè)關(guān)鍵字而不是變量小槐,每個(gè)函數(shù)調(diào)用都包含一個(gè)this值,如果閉包在外部函數(shù)里是無法訪問this的件豌,除非在外部函數(shù)將this轉(zhuǎn)存為一個(gè)變量

討論二:閉包會產(chǎn)生內(nèi)存泄漏的原因控嗜?

閉包保存在內(nèi)存里的是我們需要的變量茧彤,不屬于內(nèi)存泄漏疆栏。IE在我們使用完閉包之后,依然回收不了閉包里面引用的變量壁顶。

課后

Q1:簡述閉包是什么

A1:外部函數(shù)訪問函數(shù)內(nèi)部變量

Q2:怎么樣避免被回收

A2:將函數(shù)賦值給變量。

Q3:內(nèi)存泄漏的原因

A3:閉包保存在內(nèi)存里的是我們需要的變量,不屬于內(nèi)存泄漏富岳。IE在我們使用完閉包之后拯腮,依然回收不了閉包里面引用的變量动壤。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末琼懊,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子启妹,更是在濱河造成了極大的恐慌醉旦,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,826評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件照瘾,死亡現(xiàn)場離奇詭異丧慈,居然都是意外死亡伊滋,警方通過查閱死者的電腦和手機(jī)笑旺,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評論 3 395
  • 文/潘曉璐 我一進(jìn)店門筒主,熙熙樓的掌柜王于貴愁眉苦臉地迎上來乌妙,“玉大人,你說我怎么就攤上這事虐沥≡笏遥” “怎么了?”我有些...
    開封第一講書人閱讀 164,234評論 0 354
  • 文/不壞的土叔 我叫張陵天试,是天一觀的道長然低。 經(jīng)常有香客問我喜每,道長雳攘,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,562評論 1 293
  • 正文 為了忘掉前任鞋真,我火速辦了婚禮沃于,結(jié)果婚禮上涩咖,老公的妹妹穿的比我還像新娘。我一直安慰自己特幔,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,611評論 6 392
  • 文/花漫 我一把揭開白布蚯斯。 她就那樣靜靜地躺著饵较,像睡著了一般。 火紅的嫁衣襯著肌膚如雪循诉。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,482評論 1 302
  • 那天狈蚤,我揣著相機(jī)與錄音,去河邊找鬼脆侮。 笑死勇劣,一個(gè)胖子當(dāng)著我的面吹牛靖避,可吹牛的內(nèi)容都是我干的比默。 我是一名探鬼主播,決...
    沈念sama閱讀 40,271評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼粘咖,長吁一口氣:“原來是場噩夢啊……” “哼蚣抗!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起钝域,我...
    開封第一講書人閱讀 39,166評論 0 276
  • 序言:老撾萬榮一對情侶失蹤锭魔,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后迷捧,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體胀葱,經(jīng)...
    沈念sama閱讀 45,608評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡笙蒙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,814評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了捅位。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,926評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡尿扯,死狀恐怖焰雕,靈堂內(nèi)的尸體忽然破棺而出衷笋,到底是詐尸還是另有隱情淀散,我是刑警寧澤,帶...
    沈念sama閱讀 35,644評論 5 346
  • 正文 年R本政府宣布慢蜓,位于F島的核電站,受9級特大地震影響晨抡,放射性物質(zhì)發(fā)生泄漏则剃。R本人自食惡果不足惜耘柱,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,249評論 3 329
  • 文/蒙蒙 一棍现、第九天 我趴在偏房一處隱蔽的房頂上張望己肮。 院中可真熱鬧谎僻,春花似錦艘绍、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,866評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽堪伍。三九已至帝雇,卻和暖如春尸闸,著一層夾襖步出監(jiān)牢的瞬間孕锄,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,991評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留调卑,地道東北人恬涧。 一個(gè)月前我還...
    沈念sama閱讀 48,063評論 3 370
  • 正文 我出身青樓溯捆,卻偏偏與公主長得像提揍,于是被迫代替她去往敵國和親碳锈。 傳聞我的和親對象是個(gè)殘疾皇子欺抗,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,871評論 2 354

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