前端開發(fā)工程師必備系列-理解神奇的JS閉包

對(duì)于JavaScript新手來說瞬欧,閉包(Closures)是一個(gè)很神奇的東西。這篇講解將通過一個(gè)非常淺顯的代碼示例來解釋閉包阅羹。

計(jì)數(shù)器

我們的目標(biāo)是實(shí)現(xiàn)一個(gè)計(jì)數(shù)器为居,它的效果如下:

可知,每次執(zhí)行increment()都會(huì)輸出"Number of events: N"忌怎,且N每次都會(huì)加1籍滴。

這個(gè)計(jì)數(shù)器最直觀的實(shí)現(xiàn)方式如下:

多個(gè)計(jì)數(shù)器

以上的代碼非常簡單。但是榴啸,當(dāng)我們需要第二個(gè)計(jì)數(shù)器時(shí)孽惰,就會(huì)遇到問題了。當(dāng)然鸥印,我們可以實(shí)現(xiàn)兩個(gè)重復(fù)的計(jì)數(shù)器:

顯然勋功,以上的代碼非常冗余,有待優(yōu)化库说。當(dāng)我們需要更多計(jì)數(shù)器時(shí)狂鞋,使用這種方法將不太現(xiàn)實(shí)。這時(shí)潜的,就需要神奇的閉包了骚揍。

使用閉包實(shí)現(xiàn)計(jì)數(shù)器

需要多個(gè)計(jì)數(shù)器,同時(shí)希望去除冗余代碼的話啰挪,就可以使用閉包了:

在代碼中信不,我們創(chuàng)建了兩個(gè)獨(dú)立的計(jì)數(shù)器counter1counter2纤掸,分別進(jìn)行計(jì)數(shù),互不干撓浑塞。代碼看著有點(diǎn)奇怪借跪,我們不妨拆分起來分析。

首先酌壕,我們來看看createCounter

創(chuàng)建了一個(gè)局部變量counter

創(chuàng)建了一個(gè)局部函數(shù)increment()掏愁,它可以對(duì)counter變量進(jìn)行加1操作。

將局部函數(shù)increment()返回卵牍。注意果港,返回的是函數(shù)本身,而不是函數(shù)調(diào)用的結(jié)果糊昙。

看起來辛掠,createCounter()函數(shù)與我們最初定義的計(jì)數(shù)器非常相似。唯一的不同點(diǎn)在于:createCounter()將計(jì)數(shù)器封裝在一個(gè)函數(shù)內(nèi)释牺,于是我們將它稱作閉包萝衩。

難以理解的一點(diǎn)在于,當(dāng)我們使用createCounter()函數(shù)創(chuàng)建計(jì)數(shù)器時(shí)没咙,實(shí)際上創(chuàng)建了一個(gè)新的函數(shù):

閉包的神奇之處在于猩谊。每次使用createCounter()函數(shù)創(chuàng)建計(jì)數(shù)器increment時(shí),都會(huì)創(chuàng)建一個(gè)對(duì)應(yīng)的counter變量祭刚。并且牌捷,返回的increment函數(shù)會(huì)始終記住counter變量

更重要的是涡驮,這個(gè)counter變量是相互獨(dú)立的暗甥。比如,當(dāng)我們創(chuàng)建2個(gè)計(jì)數(shù)器時(shí)捉捅,每個(gè)計(jì)數(shù)器都會(huì)創(chuàng)建一個(gè)新的counter變量:

為計(jì)數(shù)器命名

多個(gè)計(jì)數(shù)器的輸出信息都是“Number of events: N”撤防,這樣容易混淆。如果可以為每個(gè)計(jì)數(shù)器命名锯梁,則更加方便:

通過給createCounter傳遞一個(gè)新的counterName參數(shù)即碗,可以很容易地做到這一點(diǎn):

這樣焰情,createCounter()函數(shù)返回的計(jì)數(shù)器將同時(shí)記住兩個(gè)局部變量:counterNamecounter陌凳。

優(yōu)化計(jì)數(shù)器調(diào)用方式

按照之前的實(shí)現(xiàn)方式,我們通過調(diào)用createCounter()函數(shù)可以返回一個(gè)計(jì)數(shù)器内舟,直接調(diào)用返回的計(jì)數(shù)器就可以加1合敦,這樣做并不直觀。如果可以如下調(diào)用將更好:

實(shí)現(xiàn)代碼:

可看出验游,以上的代碼返回了一個(gè)對(duì)象充岛,這個(gè)對(duì)象包含了一個(gè)increment方法保檐。

添加decrement方法

現(xiàn)在,我們可以給計(jì)數(shù)器添加一個(gè)decrement()方法

添加私有方法

前面的代碼有兩行重復(fù)的代碼崔梗,即console.log語句夜只。因此,我們可以創(chuàng)建一個(gè)display()方法用于打印counter的值:

看起來蒜魄,display()函數(shù)與increment()函數(shù)以及decrement()函數(shù)差不多扔亥,但是其實(shí)它們很不一樣。我們并沒有將display()函數(shù)添加到返回的對(duì)象中谈为,這就意味著以下代碼會(huì)出錯(cuò):

這時(shí)旅挤,display()相當(dāng)于一個(gè)私有方法,我們只能在createCounter()函數(shù)內(nèi)使用它伞鲫。

閉包與面向?qū)ο缶幊?/b>

如果你接觸過面向?qū)ο缶幊?OOP)粘茄,則應(yīng)該不難發(fā)現(xiàn)本文中所涉及的內(nèi)容與OOP中的對(duì)象秕脓、對(duì)象屬性柒瓣、共有方法私有方法等概念非常相似。

閉包吠架,與OOP相似嘹朗,就是把數(shù)據(jù)和操作數(shù)據(jù)的方法綁定起來。因此诵肛,在需要OOP的時(shí)候屹培,就可以使用閉包來實(shí)現(xiàn)。

總結(jié)

閉包(Closure)是JavaScript一個(gè)非常棒的特性怔檩。掌握它褪秀,我們可以從容應(yīng)對(duì)一些常見的編程需求。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末薛训,一起剝皮案震驚了整個(gè)濱河市媒吗,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌乙埃,老刑警劉巖闸英,帶你破解...
    沈念sama閱讀 219,427評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異介袜,居然都是意外死亡甫何,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門遇伞,熙熙樓的掌柜王于貴愁眉苦臉地迎上來辙喂,“玉大人,你說我怎么就攤上這事∥『模” “怎么了秋麸?”我有些...
    開封第一講書人閱讀 165,747評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長炬太。 經(jīng)常有香客問我灸蟆,道長,這世上最難降的妖魔是什么亲族? 我笑而不...
    開封第一講書人閱讀 58,939評(píng)論 1 295
  • 正文 為了忘掉前任次乓,我火速辦了婚禮,結(jié)果婚禮上孽水,老公的妹妹穿的比我還像新娘票腰。我一直安慰自己,他們只是感情好女气,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評(píng)論 6 392
  • 文/花漫 我一把揭開白布杏慰。 她就那樣靜靜地躺著,像睡著了一般炼鞠。 火紅的嫁衣襯著肌膚如雪缘滥。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,737評(píng)論 1 305
  • 那天谒主,我揣著相機(jī)與錄音朝扼,去河邊找鬼。 笑死霎肯,一個(gè)胖子當(dāng)著我的面吹牛擎颖,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播观游,決...
    沈念sama閱讀 40,448評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼搂捧,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了懂缕?” 一聲冷哼從身側(cè)響起允跑,我...
    開封第一講書人閱讀 39,352評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎搪柑,沒想到半個(gè)月后聋丝,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,834評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡工碾,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評(píng)論 3 338
  • 正文 我和宋清朗相戀三年弱睦,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片倚喂。...
    茶點(diǎn)故事閱讀 40,133評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡每篷,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出端圈,到底是詐尸還是另有隱情焦读,我是刑警寧澤,帶...
    沈念sama閱讀 35,815評(píng)論 5 346
  • 正文 年R本政府宣布舱权,位于F島的核電站矗晃,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏宴倍。R本人自食惡果不足惜张症,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望鸵贬。 院中可真熱鬧俗他,春花似錦、人聲如沸阔逼。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,022評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽嗜浮。三九已至羡亩,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間危融,已是汗流浹背畏铆。 一陣腳步聲響...
    開封第一講書人閱讀 33,147評(píng)論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留吉殃,地道東北人辞居。 一個(gè)月前我還...
    沈念sama閱讀 48,398評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像蛋勺,于是被迫代替她去往敵國和親速侈。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評(píng)論 2 355

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理迫卢,服務(wù)發(fā)現(xiàn)倚搬,斷路器,智...
    卡卡羅2017閱讀 134,669評(píng)論 18 139
  • 閉包 在本文章中** 閉包是指那些能夠訪問獨(dú)立(自由)變量的函數(shù) (變量在本地使用乾蛤,但定義在一個(gè)封閉的作用域中)每界。...
    __Seve閱讀 518評(píng)論 0 0
  • 第2章 基本語法 2.1 概述 基本句法和變量 語句 JavaScript程序的執(zhí)行單位為行(line),也就是一...
    悟名先生閱讀 4,150評(píng)論 0 13
  • 2017-03-22星期三"育心麗謙時(shí)間管理100天挑戰(zhàn)營” 第64天 A家卖、今日行動(dòng)清單 1眨层、【早起匯報(bào) 】: 4...
    嘉雁38677閱讀 91評(píng)論 1 2
  • 歲月未老人己老, 堂前花落知多少上荡。 月下庭院深幽靜趴樱, 邂逅梅花枝頭俏馒闷。 品茶吟詩悟人生, 笑看瑞雪空中飄叁征。 秋去冬...
    清風(fēng)明月馮耀杰閱讀 295評(píng)論 1 3