亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

JavaScript閉包詳解

2019-11-20 13:14:58
字體:
來源:轉載
供稿:網友

在上一篇文章我們對預解釋作了概述,在寫這篇博文前打算寫幾個經典案例,考慮到那些案例綜合性比較強,也就循序漸進的有了這篇博文,這樣對于學習和深入JavaScript也更加容易入手。

一同事去面試,面試官問了一道題:你寫一個閉包我看下?于是同事火速寫出如下代碼:

復制代碼 代碼如下:

function fn(){
    alert('Hello JavaScript Closure!!!');//媽蛋,E文本來就不好,找翻譯才把閉包單詞寫出來
}
fn();

然后面試官搖搖頭說道:“這怎么能叫閉包呢?”,最終兩人爭執不下,同事果斷走人,面試官什么玩意兒?(本故事純屬虛構,如有雷同純屬巧合)

閉包可能在很多人眼中都是“高大不好上”的技術,可能在很多人眼中只有這樣才算得上閉包:

示例1:

復制代碼 代碼如下:

function fn() {
    return function () {
        alert('示例1');
    }
}
fn()();

示例1 PS:這個看起來不怎么高級,看樣子這人水平不咋地哦!

示例2:

復制代碼 代碼如下:

;(function () {
    alert('示例2');
})();

示例2 PS:這個看起來比上一個要高級,而且第一個括號前還加了一個分號,為何加一個分號,好吧我們先把這個疑問留這兒,后面會講到。

示例3:

復制代碼 代碼如下:

~function fn() {
    alert('示例3')
}();

示例3 PS:這個最高級了,簡直吊炸天,我讀書少,你們別騙我!

擼主讀書不多,僅能寫出這三種“閉包”,相信博友們能寫出更多更優秀的“閉包”;到此請先暫停我的瞎掰,接下來研究下函數運行的機制,貌似有人已經知道了,肯定是作用域,我真的很不想在標題上再加上這個作用域,這樣總感覺差點兒意思,這個幾個東西本來都是一起的,為何要重復呢?老習慣,先上代碼:

復制代碼 代碼如下:

var n = 10;
function fn(){
    alert(n);
    var n = 9;
    alert(n);
}
fn();

好簡單的說,我們畫圖(擼主只會用Windows自帶的畫圖軟件,若有更好的請博友推薦)來分析下:

分析1

    從圖中我們看到了兩個作用域,一個是window作用域(頂級作用域),一個是fn調用的時候形成的一個私有作用域;那什么是作用域,作用域其實就是代碼執行的環境。舉個栗子,一個學生他的學習環境是學校,相當于他的作用域是學校,假如這個學生很調皮,晚上經常FanQiang去網吧打游戲,相當于形成了一個私有環境,這個作用域就是網吧。好吧!這個栗子太TM像擼主本人了,不由感嘆一句:“少壯不努力,長大干挨踢”。還是回到正題,其實函數fn的定義就是指向一段代碼的描述(圖中紅框),當這個fn調用(圖中的綠框)的時候,就會形成一個作用域,當然這個作用域中的代碼執行前也會預解釋,我是不會告訴你這個作用域是當它執行完畢后會被銷毀,這個fn再次調用也會形成一個新的作用域,然后執行前預解釋,然后代碼執行,最后執行完畢銷毀。

理解閉包

    我們知道函數被調用在執行的時候會形成一個私有作用域(執行環境),這個私有作用域就是閉包?;仡^再看看閉包還是傳說中的“高大不好上”嗎?我們再回頭看看第一個面試故事,還有我寫的三個示例,它們其實都是閉包,確切的說那三個示例都是閉包的常用形式。

應用場景

現在有這樣一個需求:HTML頁面中有一個ul標簽,ul下面有5個li標簽,要求任意點擊一個li,彈出被點擊的這個li所在的索引(索引從0開始)位置,HTML結構如下:

復制代碼 代碼如下:

<ul id="ul">
    <li>列表1</li>
    <li>列表2</li>
    <li>列表3</li>
    <li>列表4</li>
    <li>列表5</li>
</ul>

機智的我火速寫出如下代碼:

復制代碼 代碼如下:

var lis = document.getElementById('ul').getElementsByTagName('li');
for (var i = 0, len = lis.length; i < len; i++) {
    lis[i].onclick = function () {
        alert(i);
    };
}

最終測試,看是否完美實現這個需求:

發現無論點擊多少次,最終都彈出這個結果,而需求期望的結果是:點擊列表1彈出0,點擊列表2彈出1,點擊列表3彈出2……此時此刻只想用這幅圖來形容現在的心情:

(當原型在演示時沒能按設計的要求運行時的樣子)

這可如何才好,為何總是彈出5呢?理論上很正確呀!我們不妨畫圖來分析下:

其實我們只是給每一個li的onclick其實就是保存的一段函數的描述字符串,這個字符串內容就是上圖紅框中的內容,如果您還是不信,我有圖有真相:

在Chrome控制臺下輸入:lis[4].onclick,其值就是函數的描述。當我們在點擊第5個列表時,其實就是相當于lis[4].onclick(),調用了這段函數描述,我們知道函數在被調用執行的時會形成一個私有作用域,在這個私有作用域下也是先預解釋,然后代碼執行,此時會去找i,在當前私有作用域下沒有i,然后去window作用域下找到了i,因此每次點擊都彈出5。

顯然上面的代碼無法滿足這個需求,我們代碼那么寫是不正確的,我們思考一下出現問題的原因是什么?其實原因就是每次點擊的時候都是讀取的window下的i,此時這個i的值已經是5了,于是有了如下代碼:

方式一:

復制代碼 代碼如下:

var lis = document.getElementById('ul').getElementsByTagName('li');
function fn(i) {
    return function () {
        alert(i);
    }
}
for (var i = 0, len = lis.length; i < len; i++) {
    lis[i].onclick = fn(i);
}

方式二:

復制代碼 代碼如下:

var lis = document.getElementById('ul').getElementsByTagName('li');
 
for (var i = 0, len = lis.length; i < len; i++) {
    ;(function (i) {
        lis[i].onclick = function () {
            alert(i);
        };
    })(i);
}

方式三:

復制代碼 代碼如下:

var lis = document.getElementById('ul').getElementsByTagName('li');
 
for (var i = 0, len = lis.length; i < len; i++) {
    lis[i].onclick = function fn(i) {
        return function () {
            alert(i);
        }
    }(i);
}

一口氣寫了三種方式,其思想都是一樣的,就是將這個變量i用一個私有變量存儲起來,這里我就只講方式二,當然明白其中一個其余也就都明白了。按照慣例,我們畫圖來一步步分析下:

我詳細的對整個代碼執行做了描述,需要注意的是:每個li的onclick屬性都要占用(function(i){ … })(i)作用域,當這個函數執行完畢后不會被銷毀,因為它被外面的li(這個li是window作用域下的)占用著,因此這個作用域不會被銷毀。當點擊任意一個li時,function(){ alert(i); }會被執行,也會形成一個作用域,這個作用域沒有i,它會去(function(){ … })(i)作用域找i,最終在形參找到i,這個形參i的值就是for循環時傳進去的;這個例子巧妙地使用閉包來貯存值,完美解決問題。

PS:剛剛說(function(i){ … })(i)為什么在前面加一個分號,其原因就是防止前面的語句忘記加分號,這樣導致JavaScript在解析時出錯,僅此而已。當然上面的一個應用場景就是Tabs實現原理,可以有其他實現方式,比如自定義屬性方式、通過DOM節點關系找到索引,而擼主采用這樣一種方式只是為了加深對閉包的理解。

總結

    閉包并不是傳說中的高大不好上,其核心就是理解函數定義、調用,函數調用時會形成一個新的私有作用域,當某個作用域被外面占用,那么這個作用域將不會被銷毀。擼主讀書甚少,有說得不對的地方請博友們指正,同時也感謝大家對擼主文章的支持。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲激情在线观看视频免费| 亚洲电影免费观看高清完整版在线观看| 欧美成人精品影院| 欧美国产精品日韩| 久久久99免费视频| 久久免费福利视频| 亚洲影视中文字幕| 国产成人一区二区| 久久国产精彩视频| 激情亚洲一区二区三区四区| 国产精品视频网站| 欧美黑人一区二区三区| 97视频在线观看播放| 91精品视频大全| 日韩中文字幕精品| 九色精品美女在线| 精品高清一区二区三区| 亚洲欧美精品在线| 92国产精品久久久久首页| 亚洲人成电影在线观看天堂色| 色综合导航网站| 97超视频免费观看| 亚洲一区二区三区xxx视频| 91国自产精品中文字幕亚洲| 欧美激情喷水视频| 国产高清在线不卡| 5278欧美一区二区三区| 国产精品久久久久高潮| 欧美老女人性生活| 自拍亚洲一区欧美另类| 久久激情视频久久| 国产精品久久久久久久久影视| 国产一区二区三区在线播放免费观看| 视频在线观看99| 欧美三级欧美成人高清www| 国产高清视频一区三区| 国产日韩精品入口| 九九综合九九综合| www.精品av.com| 国产精品美女主播在线观看纯欲| 欧美精品videosex牲欧美| 欧美精品久久久久a| 狠狠久久亚洲欧美专区| 亚洲日本成人网| 1769国产精品| 亚洲精品久久久久久久久| 亚洲性日韩精品一区二区| 亚洲天堂视频在线观看| 欧美人与物videos| 欧美日韩国产中文精品字幕自在自线| 欧美精品一二区| 久久久久国产精品免费| 欧美日韩xxx| 96pao国产成视频永久免费| 久久网福利资源网站| 欧洲亚洲免费视频| 日韩暖暖在线视频| 久久久免费av| 久久久久久网址| 国产欧美亚洲精品| 欧美激情亚洲国产| 欧美成人午夜免费视在线看片| 激情成人中文字幕| 国产精品免费观看在线| 精品亚洲一区二区三区在线播放| 中文日韩在线观看| 亚洲精品电影在线| 91麻豆国产精品| 久久综合国产精品台湾中文娱乐网| 国产一区二区精品丝袜| 91香蕉嫩草影院入口| 国语对白做受69| 欧美美最猛性xxxxxx| 欧美风情在线观看| 免费91麻豆精品国产自产在线观看| 亚洲美女av在线| 国产精品亚洲一区二区三区| 国产精彩精品视频| 久久av红桃一区二区小说| 国产精品免费久久久久久| 日韩大陆欧美高清视频区| 久久6精品影院| 国产极品jizzhd欧美| 欧美精品激情blacked18| 成人免费视频xnxx.com| 欧美精品video| 日韩电影大片中文字幕| 成人中文字幕在线观看| 国产精品视频一区二区高潮| 久久久999国产精品| 伊人久久大香线蕉av一区二区| 91热福利电影| 国产精品久久久久久av福利| 国产精品99蜜臀久久不卡二区| 久久国产精品视频| 欧美亚洲另类制服自拍| 奇米4444一区二区三区| 欧美日韩精品在线播放| 欧美高清videos高潮hd| 国产精品91免费在线| 日韩电影中文字幕在线观看| 久久久久久久久国产精品| 久久天天躁狠狠躁夜夜爽蜜月| 久久影院资源站| 日韩免费在线视频| 久久久久久国产精品| 欧美—级a级欧美特级ar全黄| 国产aaa精品| 日韩精品在线视频美女| 色视频www在线播放国产成人| 91人成网站www| 久久成人人人人精品欧| 国产精品一区二区久久久| 九九热99久久久国产盗摄| 午夜免费日韩视频| 岛国精品视频在线播放| 国产亚洲人成网站在线观看| 欧美日韩在线另类| 91色琪琪电影亚洲精品久久| 亚洲三级黄色在线观看| 欧美三级欧美成人高清www| 日韩av综合网站| 国产成人涩涩涩视频在线观看| 国产成人aa精品一区在线播放| 国产精品∨欧美精品v日韩精品| 正在播放亚洲1区| 国产美女精品免费电影| 国产精品v日韩精品| 欧美激情国产精品| 欧美激情第6页| 国产欧美一区二区| 亚洲高清一区二| 欧美另类极品videosbest最新版本| 欧美一级黑人aaaaaaa做受| 亚洲精美色品网站| 精品国产一区二区三区久久狼黑人| 97精品伊人久久久大香线蕉| 国产精品日韩欧美大师| 欧美日韩中文在线观看| 亚洲片国产一区一级在线观看| 精品无人国产偷自产在线| 成人黄色av网| 中文字幕少妇一区二区三区| 日韩精品在线播放| 日韩h在线观看| 狠狠躁夜夜躁久久躁别揉| 欧洲中文字幕国产精品| 亚洲性av在线| www.亚洲男人天堂| 国产精品免费看久久久香蕉| 日韩精品在线免费观看视频| 高清欧美性猛交xxxx黑人猛交| 日韩精品亚洲精品| 欧美午夜精品伦理| 国产日韩精品入口| 中文字幕在线日韩| 欧美黑人视频一区| 92国产精品视频| 在线播放国产精品| 亚洲国产精品电影在线观看| 欧美精品在线视频观看| 久久视频在线看| 亚洲第一综合天堂另类专| 日韩a**站在线观看|