理解JavaScript中的閉包

閉包并不復雜。學習和理解閉包的基礎知識僅僅只需要10分鐘。

什么是閉包?

閉包是JavaScript的一部分,每個工程師都應該知道和理解閉包。今天的文章只是撇開閉包的表面,但是將會給你一個很好的思想,關于什么是閉包和它在JavaScript中是如何運行的。好了,接下來我們開始吧。

我們先從書本中閉包的定義開始。

定義1

閉包是一個能訪問父作用域的函數,即使此作用域關閉了。

定義2

閉包是一個函數和該函數被聲明的詞匯環境的組合。

很好,但是這些真正的意思是什么?

首先你需要理解在JavaScript中的作用域,作用域實質上是變量在JavaScript中的生命周期。因此,當一個變量定義了一個大的作用域,這個變量會存活多久,那個方法在你的程序中會訪問到它。

讓我們看一個例子。

當你在JavaScript中創建一個函數,它可以訪問創建在這個函數里面和外面的變量。

變量創建在一個函數中也定義了這個變量。一個局部變量只能在這個被定義的函數里面訪問到。在下面這個例子中,你將會看到如果我們嘗試在這個函數外面打印words,它會輸出錯誤信息。這是因為words是一個內部作用域。

function speak () {
  var words = 'hi';
  console.log(words);
}
speak(); // hi
console.log(words); // undefined

和上一個例子不同,這次我們定義這個words 為全局作用域。
意思是說我們可以在任何函數中訪問這個變量。

// 將變量放到函數的外面
// words 現在是個全局變量
var words = 'hi';
function speak(){ 
  console.log(words);
}
speak(); // 'hi' 
console.log(words); // 'hi'

嵌套函數

我們將一個函數放到另一個函數里面會發生什么?我想你跟著操作下面的例子,因為這個會很有趣!

如果你使用谷歌瀏覽器,打開你的開發工具調試模式:
[Windows]:Ctrl + Shift + J
[MAC]:Cmd + Opt + J

Cool,現在拷貝下面的代碼然后粘貼到你的控制臺中。我們現在做的是創建一個名字為speak的函數。speak返回一個名稱為logIt的函數。最終logIt在控制臺中打印log值為words,在這個實例要實現在控制臺中輸出 ‘hi’

function speak() {
  return function logIt() {
    var words = 'hi';
    console.log(words);
  }
}

當你拷貝到這段代碼到你的控制臺中,我們將要創建一個變量,然后將speak函數賦值給它。

var sayHello = speak();

現在我們可以看見這個變量sayHello 調用了之后,沒有執行內部函數。

sayHello;
//  function logIt() {
//    var words = 'hi';
//    console.log(words);
//  }

如上打印的結果,sayHello 打印的是我們return的內部函數,也就是說,如果我們執行sayHello()在控制臺中,它將會喚起執行logIt()函數:

sayHello();
// 'hi'

它奏效了,但是并沒有任何特別。讓我們移除一行代碼看看什么發生了改變。請看下面的示例。我們將清楚定義的變量words移到speak()函數的里面。

function speak() {
  var words = 'hi';
  return function logIt() {
    console.log(words);
  }
}

像之前的操作,讓我們定義一個變量并將speak函數賦值給它:

var sayHello = speak();

現在我們看看我們的 sayHello變量會輸出什么:

sayHello
//  function logIt() {
//    console.log(words);
//  }

哦,這里沒有words 變量的定義,那么發生了什么當我們執行這個函數的時候?

sayHello();
// 'hi'

它仍然起作用了!這是因為你剛剛體驗了閉包的影響。

是否有點疑惑?沒關系,回想下我們的閉包的定義:

閉包是一個能訪問父作用域的函數,即使此作用域關閉了。

在這個例子中我們的speak()函數作用域關閉了。這意味著 var words = 'hi' 應該消失了。然而,在JS中我們有一個名稱將這種現象叫做閉包:我們的內部函數保持對其創建的范圍的引用。這就允許logIt()函數仍然可以訪問words這個變量——即使speak()這個函數作用域關閉了。

function speak() {
  var words = 'hi';
  return function logIt() {
    console.log(words);
  }
}

重要提示,在JavaScript中每個函數都有一個閉包。你不需要你去解釋閉包在函數中是怎么運行的,它僅僅只是JavaScript的一部分。

實例2

讓我們查看另一個例子。這個例子有點復雜,代碼如下:

function name(n) {
  return function(a) {
    return `${n} likes ${a}`;
  };
}

我們定義了一個名字為name 的函數可以傳遞一個參數,返回一個匿名函數,可以傳遞一個不同的參數,內部函數返回一個字符串。

我們用name函數創建兩個變量。一個我們給name函數傳遞 John,另一個傳 Cindy:

var j = name('John');
var c = name('Cindy');

讓我們看看j現在能打印什么:

j;
//  function (a) {
//    return `${n} likes ${a}`;
//  }

所以根據之前示例我們知道這是因為閉包,這個函數應該仍然可以訪問n的變量通過父作用域。我們所能做的是把a傳遞過去,在執行函數的時候。

讓我們試試:

j('dogs');  // 'John likes dogs'
c('cats');  // 'Cindy likes cats'

成功了!因為閉包我們才能從之前關閉的作用域引用變量成功的執行我們的函數。

總結

希望你現在可以理解基本的閉包在JavaScript和它是如何運行的!這雖然只是冰山一角,但明白了基礎知識才能學習和練習更復雜的閉包。


翻譯原文鏈接:https://codeburst.io/understand-closures-in-javascript-d07852fa51e7
翻譯中有疏漏的地方,歡迎指正。

校對:蹦蹦

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,362評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,013評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,346評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,421評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,146評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,534評論 1 325
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,585評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,767評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,318評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,074評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,258評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,828評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,486評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,916評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,156評論 1 290
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,993評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,234評論 2 375

推薦閱讀更多精彩內容