隨手記一記之【函數(shù)表達(dá)式與閉包】

參考

定義函數(shù)

定義函數(shù)有兩種方式:一是函數(shù)聲明踱侣,而是函數(shù)表達(dá)式购公。

函數(shù)聲明重要的特征是织阅,函數(shù)聲明提升——在執(zhí)行代碼之前會先讀取函數(shù)聲明,這就意味著可以把函數(shù)聲明放在調(diào)用此函數(shù)的語句后面雕擂。

sayHi()

function sayHi() {
    console.log('hi')
}

// hi

然而踊东,函數(shù)表達(dá)式?jīng)]有這種特性,比如說:

sayHi()

const sayHi = function() {
  console.log('hi')
}

//error: Uncaught TypeError: sayHi is not a function

閉包

定義

閉包是指有權(quán)訪問另一個函數(shù)作用域中的變量的函數(shù) ——JavaScript 高級程序設(shè)計(第三版)

看了紅寶書的閉包胞四,和很多關(guān)于閉包的文章恬汁。覺得還是紅寶書的描述規(guī)范一點,怎么理解這句話呢辜伟?簡單來說就是函數(shù)能訪問別的函數(shù)的變量氓侧,這個函數(shù)就是閉包。

把閉包簡單理解成"定義在一個函數(shù)內(nèi)部的函數(shù)"导狡≡枷铮—— 學(xué)習(xí)Javascript閉包(Closure)——(阮一峰)

閉包用途——讀取函數(shù)內(nèi)部的變量

function A() {
  var value = 999
  function B() {
    console.log('value', value)
  }
  B()
}
A() //999

函數(shù) B 在函數(shù) A中定義,那么函數(shù) B 可以訪問 A 中的變量(還可以訪問全局變量)旱捧。函數(shù) B 就是閉包独郎,很簡單吧。再看:

function A() {
  var value = 999
}
function B() {
  console.log('value', value)
}
B()
//報錯: Uncaught ReferenceError: value is not defined

此時,函數(shù) B 不能訪問函數(shù) A 中的 value囚聚,B就不是閉包靖榕。就是這樣子

經(jīng)常會看到用兩個括號的方式使用閉包,例如:

(function(value) {
  function B() {
    console.log(value)
  }
  B()
})(999)

// 999

以前一直不明白這是什么意思顽铸,其實仔細(xì)看下茁计。就是一個立即執(zhí)行的匿名函數(shù),匿名函數(shù)中的函數(shù) B 能訪問匿名函數(shù)的變量(參數(shù))谓松,函數(shù) B 就是閉包星压。

閉包用途—— 讓變量的值始終保持在內(nèi)存中

function f1() {
  var name = 'MyName'
  function sayHello() {
    alert('hello ' + name)
  }

  sayHello()
}
f1() // hello MyName

這段代碼,當(dāng) f1執(zhí)行后鬼譬,變量 name就會回收娜膘。再看下面:

function f1() {

  var name = 'MyName'

  function sayHello() {
    alert('hello ' + name)
  }

  return sayHello

}

var result = f1()

result() // hello MyName

函數(shù) f1 返回了函數(shù) sayHello,并保存在全局變量 result 中,這里 result 就是 sayHello 函數(shù)閉包优质。無論調(diào)用多少次 result 都能訪問 f1函數(shù)內(nèi)的變量 name竣贪,這就意味著name 一直在內(nèi)存中,并沒有在f1執(zhí)行后被回收巩螃。

經(jīng)典面試題

要求每隔一秒打印 1演怎、2、3避乏、4爷耀、5,很容易寫出這樣的代碼來:

for (var i = 1; i <= 5; i++) {
     setTimeout(function timer() {
        console.log(i)
      }, 1000 * i)
    }

結(jié)果是每隔一秒打印了一個6出來拍皮。由于setTimeout是異步的歹叮,首先會把循環(huán)執(zhí)行完。等到打印時铆帽,變量 i 已經(jīng)是6了咆耿,于是打印了5個6。

解決辦法:閉包

    for (var i = 1; i <= 5; i++) {
      (function(j) {
        setTimeout(function timer() {
          console.log(j)
        }, j * 1000)
      })(i)
    }

這里為什么就不是打印6呢锄贼?timer就是閉包票灰,time 訪問的是 匿名函數(shù)的參數(shù) j 。相當(dāng)于循環(huán)里面調(diào)用了5次匿名函數(shù)宅荤,每次傳給匿名函數(shù)的參數(shù)分別是1、2浸策、3冯键、4、5庸汗”谷罚可以理解為每次調(diào)用匿名函數(shù),都給其參數(shù) j 賦值 i。所以最終可以正確打印1改化、2掩蛤、3、4陈肛、5揍鸟。

解決這個問題還要其他的辦法。
方法一:setTimeout 第三個參數(shù)

for (var i = 1; i <= 5; i++) {
  setTimeout(function(j) {
    console.log(j)
  }, i * 1000, i)
}

方法二: 用 let 聲明變量

    for (let i = 1; i <= 5; i++) {
      setTimeout(function() {
        console.log(i)
      }, i * 1000)
    }
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末句旱,一起剝皮案震驚了整個濱河市阳藻,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌谈撒,老刑警劉巖腥泥,帶你破解...
    沈念sama閱讀 212,383評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異啃匿,居然都是意外死亡蛔外,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評論 3 385
  • 文/潘曉璐 我一進(jìn)店門溯乒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來冒萄,“玉大人,你說我怎么就攤上這事橙数∽鹆鳎” “怎么了?”我有些...
    開封第一講書人閱讀 157,852評論 0 348
  • 文/不壞的土叔 我叫張陵灯帮,是天一觀的道長崖技。 經(jīng)常有香客問我,道長钟哥,這世上最難降的妖魔是什么迎献? 我笑而不...
    開封第一講書人閱讀 56,621評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮腻贰,結(jié)果婚禮上吁恍,老公的妹妹穿的比我還像新娘。我一直安慰自己播演,他們只是感情好冀瓦,可當(dāng)我...
    茶點故事閱讀 65,741評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著写烤,像睡著了一般翼闽。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上洲炊,一...
    開封第一講書人閱讀 49,929評論 1 290
  • 那天感局,我揣著相機與錄音尼啡,去河邊找鬼。 笑死询微,一個胖子當(dāng)著我的面吹牛崖瞭,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播撑毛,決...
    沈念sama閱讀 39,076評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼书聚,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了代态?” 一聲冷哼從身側(cè)響起寺惫,我...
    開封第一講書人閱讀 37,803評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蹦疑,沒想到半個月后西雀,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,265評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡歉摧,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,582評論 2 327
  • 正文 我和宋清朗相戀三年艇肴,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片叁温。...
    茶點故事閱讀 38,716評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡再悼,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出膝但,到底是詐尸還是另有隱情冲九,我是刑警寧澤,帶...
    沈念sama閱讀 34,395評論 4 333
  • 正文 年R本政府宣布跟束,位于F島的核電站莺奸,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏冀宴。R本人自食惡果不足惜灭贷,卻給世界環(huán)境...
    茶點故事閱讀 40,039評論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望略贮。 院中可真熱鬧甚疟,春花似錦、人聲如沸逃延。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽真友。三九已至黄痪,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間盔然,已是汗流浹背桅打。 一陣腳步聲響...
    開封第一講書人閱讀 32,027評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留愈案,地道東北人挺尾。 一個月前我還...
    沈念sama閱讀 46,488評論 2 361
  • 正文 我出身青樓,卻偏偏與公主長得像站绪,于是被迫代替她去往敵國和親遭铺。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,612評論 2 350

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