閉包一

推薦阮一峰的博客
http://www.ruanyifeng.com/blog/2009/08/learning_javascript_closures.html
還有這個(gè)人的博客(強(qiáng)烈推薦里面的東西矛辕,全面詳細(xì)=)
http://www.108js.com/search.jsp
下面是自己的一些理解

<h2>首先要理解變量作用域:</h2>

  • 全局變量
  • 函數(shù)內(nèi)部可以直接訪問(wèn)全局變量
 var n=999;

   function f1(){
     alert(n);
    }
  f1(); // 999
  • 局部變量
  • 在函數(shù)外部自然無(wú)法讀取函數(shù)內(nèi)的局部變量
 function f1(){
    var n=999;
  }

  alert(n); // error

了解了作用域后茂翔,思考一個(gè)問(wèn)題,如果想要在函數(shù)外部讀取局部變量怎么辦杈绸?
在函數(shù)內(nèi)部再定義一個(gè)函數(shù)就可以訪問(wèn)局部變量了。
<h2>從外部讀取局部變量</h2>

function f1(){

    var n=999;

    function f2(){
      alert(n); // 999
    }

  }

在上面的代碼中矮瘟,函數(shù)f2就被包括在函數(shù)f1內(nèi)部瞳脓,這時(shí)f1內(nèi)部的所有局部變量,對(duì)f2都是可見(jiàn)的澈侠。但是反過(guò)來(lái)就不行劫侧,f2內(nèi)部的局部變量,對(duì)f1就是不可見(jiàn)的哨啃。這就是Javascript語(yǔ)言特有的<b>"鏈?zhǔn)阶饔糜?</b>結(jié)構(gòu)(chain scope)烧栋,子對(duì)象會(huì)一級(jí)一級(jí)地向上尋找所有父對(duì)象的變量。所以棘催,父對(duì)象的所有變量劲弦,對(duì)子對(duì)象都是可見(jiàn)的,反之則不成立醇坝。

既然f2可以讀取f1中的局部變量邑跪,那么只要把f2作為返回值,我們不就可以在f1外部讀取它的內(nèi)部變量了嗎呼猪!

 function f1(){

    var n=999;

    function f2(){
      alert(n);
    }

    return f2;

  }

  var result=f1();

  result(); // 999

<h2>閉包</h2>
上面例子中的函數(shù)f2()就是個(gè)閉包画畅。<b>閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)</b>。

由于在Javascript語(yǔ)言中宋距,只有函數(shù)內(nèi)部的子函數(shù)才能讀取局部變量轴踱,因此可以把閉包簡(jiǎn)單理解成"定義在一個(gè)函數(shù)內(nèi)部的函數(shù)"。

所以谚赎,在本質(zhì)上淫僻,<b>閉包就是將函數(shù)內(nèi)部和函數(shù)外部連接起來(lái)的一座橋梁</b>诱篷。

<h2>閉包的用途</h2>
閉包可以用在許多地方。它的最大用處有兩個(gè)雳灵,一個(gè)是前面提到的可以讀取函數(shù)內(nèi)部的變量棕所,另一個(gè)就是讓這些變量的值始終保持在內(nèi)存中。

function f1(){

    var 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)存中拢蛋,并沒(méi)有在f1調(diào)用后被自動(dòng)清除桦他。

為什么會(huì)這樣呢?原因就在于f1是f2的父函數(shù)瓤狐,而f2被賦給了一個(gè)全局變量瞬铸,這導(dǎo)致f2始終在內(nèi)存中,而f2的存在依賴于f1础锐,因此f1也始終在內(nèi)存中嗓节,不會(huì)在調(diào)用結(jié)束后,被垃圾回收機(jī)制(garbage collection)回收皆警。

這段代碼中另一個(gè)值得注意的地方拦宣,就是"nAdd=function(){n+=1}"這一行,首先在nAdd前面沒(méi)有使用var關(guān)鍵字信姓,因此nAdd是一個(gè)全局變量鸵隧,而不是局部變量。其次意推,nAdd的值是一個(gè)匿名函數(shù)(anonymous function)豆瘫,而這個(gè)匿名函數(shù)本身也是一個(gè)閉包,所以nAdd相當(dāng)于是一個(gè)setter菊值,可以在函數(shù)外部對(duì)函數(shù)內(nèi)部的局部變量進(jìn)行操作外驱。

<h2>使用閉包需要注意事項(xiàng)</h2>
1)由于閉包會(huì)使得函數(shù)中的變量都被保存在內(nèi)存中,內(nèi)存消耗很大腻窒,所以不能濫用閉包昵宇,否則會(huì)造成網(wǎng)頁(yè)的性能問(wèn)題,在IE中可能導(dǎo)致內(nèi)存泄露儿子。解決方法是瓦哎,在退出函數(shù)之前,將不使用的局部變量全部刪除。

2)閉包會(huì)在父函數(shù)外部蒋譬,改變父函數(shù)內(nèi)部變量的值割岛。所以,如果你把父函數(shù)當(dāng)作對(duì)象(object)使用羡铲,把閉包當(dāng)作它的公用方法(Public Method)蜂桶,把內(nèi)部變量當(dāng)作它的私有屬性(private value),這時(shí)一定要小心也切,不要隨便改變父函數(shù)內(nèi)部變量的值。

關(guān)于閉包的思考題:
代碼一:

var name = "The Window";

  var object = {
    name : "My Object",

    getNameFunc : function(){
      return function(){
        return this.name;
      };

    }

  };

  alert(object.getNameFunc()());

這段代碼中腰湾,
getNameFunc: function() {//假設(shè)函數(shù)名為A
return function()/假設(shè)函數(shù)名為B/ { return this.name; };
}
在函數(shù)里面構(gòu)建函數(shù)的時(shí)候雷恃,閉包產(chǎn)生。
在函數(shù)B內(nèi)調(diào)用函數(shù)A的this.name,由于函數(shù)A沒(méi)有name屬性费坊,所以就去找全局變量name倒槐,找到了,所以返回“The Window”附井,要是沒(méi)有找到讨越,則返回“undefined”。
代碼二:

var name = "The Window";

  var object = {
    name : "My Object",

    getNameFunc : function(){
      var that = this;
      return function(){
        return that.name;
      };

    }

  };

  alert(object.getNameFunc()()); 

內(nèi)部函數(shù)可以訪問(wèn)定義它們的外部函數(shù)的參數(shù)和變量(除了this和arguments之外)
如果需要訪問(wèn)對(duì)象的name屬性的話永毅,就需要顯示的定義一個(gè)變量that來(lái)引用this把跨,而這個(gè)變量此時(shí)就指向object對(duì)象了。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末沼死,一起剝皮案震驚了整個(gè)濱河市着逐,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌意蛀,老刑警劉巖耸别,帶你破解...
    沈念sama閱讀 222,865評(píng)論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異县钥,居然都是意外死亡秀姐,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,296評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門若贮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)省有,“玉大人,你說(shuō)我怎么就攤上這事兜看∽断蹋” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 169,631評(píng)論 0 364
  • 文/不壞的土叔 我叫張陵细移,是天一觀的道長(zhǎng)搏予。 經(jīng)常有香客問(wèn)我,道長(zhǎng)弧轧,這世上最難降的妖魔是什么雪侥? 我笑而不...
    開(kāi)封第一講書人閱讀 60,199評(píng)論 1 300
  • 正文 為了忘掉前任碗殷,我火速辦了婚禮,結(jié)果婚禮上速缨,老公的妹妹穿的比我還像新娘锌妻。我一直安慰自己,他們只是感情好旬牲,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,196評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布仿粹。 她就那樣靜靜地躺著,像睡著了一般原茅。 火紅的嫁衣襯著肌膚如雪吭历。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 52,793評(píng)論 1 314
  • 那天擂橘,我揣著相機(jī)與錄音晌区,去河邊找鬼。 笑死通贞,一個(gè)胖子當(dāng)著我的面吹牛朗若,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播昌罩,決...
    沈念sama閱讀 41,221評(píng)論 3 423
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼哭懈,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了峡迷?” 一聲冷哼從身側(cè)響起银伟,我...
    開(kāi)封第一講書人閱讀 40,174評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎绘搞,沒(méi)想到半個(gè)月后彤避,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,699評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡夯辖,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,770評(píng)論 3 343
  • 正文 我和宋清朗相戀三年琉预,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蒿褂。...
    茶點(diǎn)故事閱讀 40,918評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡圆米,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出啄栓,到底是詐尸還是另有隱情娄帖,我是刑警寧澤,帶...
    沈念sama閱讀 36,573評(píng)論 5 351
  • 正文 年R本政府宣布昙楚,位于F島的核電站近速,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜削葱,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,255評(píng)論 3 336
  • 文/蒙蒙 一奖亚、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧析砸,春花似錦昔字、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 32,749評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至蛮瞄,卻和暖如春所坯,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背挂捅。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,862評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留堂湖,地道東北人闲先。 一個(gè)月前我還...
    沈念sama閱讀 49,364評(píng)論 3 379
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像无蜂,于是被迫代替她去往敵國(guó)和親伺糠。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,926評(píng)論 2 361

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

  • 閉包(closure)是Javascript語(yǔ)言的一個(gè)難點(diǎn)斥季,也是它的特色训桶,很多高級(jí)應(yīng)用都要依靠閉包實(shí)現(xiàn)。 一酣倾、變量...
    zock閱讀 1,075評(píng)論 2 6
  • ● 閉包基礎(chǔ) ● 閉包作用 ● 閉包經(jīng)典例子 ● 閉包應(yīng)用 ● 閉包缺點(diǎn) ● 參考資料 1舵揭、閉包基礎(chǔ) 作用域和作...
    lzyuan閱讀 941評(píng)論 0 0
  • 閉包(closure)是Javascript語(yǔ)言的一個(gè)難點(diǎn),也是它的特色躁锡,很多高級(jí)應(yīng)用都要依靠閉包實(shí)現(xiàn)午绳。 一、變量...
    zouCode閱讀 1,275評(píng)論 0 13
  • 閉包(closure)是JavaScript語(yǔ)言的一個(gè)難點(diǎn)映之,也是它的特色拦焚,很多高級(jí)應(yīng)用都要依靠閉包實(shí)現(xiàn)。 下面就是...
    麥田里的野望閱讀 365評(píng)論 0 0
  • 聽(tīng)“雄安新區(qū)”與世界的對(duì)話 伴隨著近些年上海杠输、深圳的快速崛起赎败,如今又有了雄安新區(qū)這個(gè)“新鮮的”小苗兒,自深圳完成了...
    麥子dee5閱讀 249評(píng)論 0 1