當前位置:
首頁 > 科技 > 前端開發工程師必備系列-理解神奇的JS閉包

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

對於JavaScript新手來說,閉包(Closures)是一個很神奇的東西。這篇講解將通過一個非常淺顯的代碼示例來解釋閉包


計數器

我們的目標是實現一個計數器,它的效果如下:

可知,每次執行increment()都會輸出"Number of events: N",且N每次都會加1

這個計數器最直觀的實現方式如下:


多個計數器

以上的代碼非常簡單。但是,當我們需要第二個計數器時,就會遇到問題了。當然,我們可以實現兩個重複的計數器:

顯然,以上的代碼非常冗餘,有待優化。當我們需要更多計數器時,使用這種方法將不太現實。這時,就需要神奇的閉包了。


使用閉包實現計數器

需要多個計數器,同時希望去除冗餘代碼的話,就可以使用閉包了:

在代碼中,我們創建了兩個獨立的計數器counter1counter2,分別進行計數,互不幹撓。代碼看著有點奇怪,我們不妨拆分起來分析。

首先,我們來看看createCounter

創建了一個局部變數counter

創建了一個局部函數increment(),它可以對counter變數進行加1操作。

將局部函數increment()返回。注意,返回的是函數本身,而不是函數調用的結果。

看起來,createCounter()函數與我們最初定義的計數器非常相似。唯一的不同點在於:createCounter()將計數器封裝在一個函數內,於是我們將它稱作閉包

難以理解的一點在於,當我們使用createCounter()函數創建計數器時,實際上創建了一個新的函數:

閉包的神奇之處在於。每次使用createCounter()函數創建計數器increment時,都會創建一個對應的counter變數。並且,返回的increment函數會始終記住counter變數

更重要的是,這個counter變數是相互獨立的。比如,當我們創建2個計數器時,每個計數器都會創建一個新的counter變數:


為計數器命名

多個計數器的輸出信息都是「Number of events: N」,這樣容易混淆。如果可以為每個計數器命名,則更加方便:

通過給createCounter傳遞一個新的counterName參數,可以很容易地做到這一點:

這樣,createCounter()函數返回的計數器將同時記住兩個局部變數:counterNamecounter


優化計數器調用方式

按照之前的實現方式,我們通過調用createCounter()函數可以返回一個計數器,直接調用返回的計數器就可以加1,這樣做並不直觀。如果可以如下調用將更好:

實現代碼:

可看出,以上的代碼返回了一個對象,這個對象包含了一個increment方法。


添加decrement方法

現在,我們可以給計數器添加一個decrement()方法


添加私有方法

前面的代碼有兩行重複的代碼,即console.log語句。因此,我們可以創建一個display()方法用於列印counter的值:

看起來,display()函數與increment()函數以及decrement()函數差不多,但是其實它們很不一樣。我們並沒有將display()函數添加到返回的對象中,這就意味著以下代碼會出錯:

這時,display()相當於一個私有方法,我們只能在createCounter()函數內使用它。


閉包與面向對象編程

如果你接觸過面向對象編程(OOP),則應該不難發現本文中所涉及的內容與OOP中的對象對象屬性共有方法私有方法等概念非常相似。

閉包,與OOP相似,就是把數據和操作數據的方法綁定起來。因此,在需要OOP的時候,就可以使用閉包來實現。


總結

閉包(Closure)是JavaScript一個非常棒的特性。掌握它,我們可以從容應對一些常見的編程需求。

如果你覺得對你有用,記得要點贊+轉發哦!


喜歡這篇文章嗎?立刻分享出去讓更多人知道吧!

本站內容充實豐富,博大精深,小編精選每日熱門資訊,隨時更新,點擊「搶先收到最新資訊」瀏覽吧!


請您繼續閱讀更多來自 WEB開發李家靖 的精彩文章:

英特爾安全漏洞事件持續發酵,谷歌稱其發布的補丁不會拖慢速度

TAG:WEB開發李家靖 |