JavaScript閉包原理與使用介紹
1. 認(rèn)識(shí)閉包
閉包有一個(gè)很經(jīng)典的場(chǎng)景:使用 for循環(huán)給上面5個(gè)按鈕綁定點(diǎn)擊事件。
<button type="button" class='button'>按鈕</button> <button type="button" class='button'>按鈕</button> <button type="button" class='button'>按鈕</button> <button type="button" class='button'>按鈕</button> <button type="button" class='button'>按鈕</button> var buttons = document.getElementsByClassName('button'); for (var i = 0; i < 5; i++) { buttons[i].onclick = function() { console.log(i+1); } }
分別點(diǎn)擊5個(gè)按鈕控制臺(tái)輸出的都是5,由于i的作用域使的問題使得代碼沒有按照預(yù)期進(jìn)行輸出。
使用閉包對(duì)代碼進(jìn)行改進(jìn)
var buttons = document.getElementsByClassName('button'); for (var i = 0; i < 5; i++) { buttons[i].onclick = (function(i){ return function(){ console.log(i+1); } }(i)) }
再分別點(diǎn)擊5個(gè)按鈕控制臺(tái)依次輸出1、2、3、4、5
2. 變量的作用域和生命周期
2.1 變量的作用域
- 作用域即是變量的作用范圍
- 使用var關(guān)鍵字聲明的變量會(huì)提升到全局,函數(shù)里面的變量只有在函數(shù)內(nèi)部能夠訪問
- 使用let和和const關(guān)鍵字聲明的變量不提升
上面的代碼中for循環(huán)中的變量i是使用var聲明的,會(huì)提升至全局,最終打印輸出的都是全局的i,而在點(diǎn)擊按鈕之前for循環(huán)已經(jīng)走完,因此輸出的都是5。
在使用閉包改進(jìn)的時(shí)候使用立即執(zhí)行函數(shù)將每次循環(huán)的i保留在立即執(zhí)行函數(shù)的內(nèi)部,最終輸出的就是正確的結(jié)果。
2.2 變量的生命周期
全局作用域的變量的生命周期是永久的,函數(shù)內(nèi)的變量在函數(shù)執(zhí)行后被回收銷毀。
產(chǎn)生閉包的時(shí)候由于函數(shù)的返回值(函數(shù))仍然引用著函數(shù)的局部變量,導(dǎo)致即使函數(shù)執(zhí)行完函數(shù)內(nèi)的變量仍然存在。閉包正是利用這一特性。
3. 閉包的概念及其作用
3.1 閉包的概念
經(jīng)過上面的講述不難發(fā)現(xiàn)閉包主要涉及變量的生命周期和作用域。形成閉包的三個(gè)必要條件:
- 函數(shù)返回值是個(gè)函數(shù)
- 被返回的函數(shù)內(nèi)引用了函數(shù)的局部變量
- 被返回的函數(shù)在外部被調(diào)用
3.2 閉包的應(yīng)用
3.2.1 保存私有變量
Javascript中沒有private關(guān)鍵字,但可以通過閉包將變量保存在函數(shù)內(nèi)部,從而達(dá)到保護(hù)變量的作用。
var getUserInfo = (function() { var userInfo = { name: 'ian', age: 21 }; return function() { return userInfo; } }()); console.log(getUserInfo()); //{name:'ian',age:21} console.log(userInfo); //Uncaught ReferenceError: userInfo is not defined
3.2.2 使用閉包實(shí)現(xiàn)節(jié)流
function throttle(fn,delay){ var timer=null; return function(){ if(!timer){ timer=setTimeout(()=>{ fn(); timer =null; },delay) } } } window.onresize= throttle(function(){ console.log('resize') },500)
到此這篇關(guān)于JavaScript閉包原理與使用介紹的文章就介紹到這了,更多相關(guān)JS閉包內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
next.js初始化參數(shù)設(shè)置getServerSideProps應(yīng)用學(xué)習(xí)
這篇文章主要為大家介紹了next.js初始化參數(shù)設(shè)置getServerSideProps的應(yīng)用示例學(xué)習(xí),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10javascript 獲取模態(tài)窗口的滾動(dòng)位置代碼
模態(tài)窗口的滾動(dòng)位置獲取辦法還有不知道的嗎?下面的方法或許對(duì)大家有所幫助,感興趣的朋友可以了解下,希望對(duì)大家有所幫助2013-08-08JavaScript選擇排序算法原理與實(shí)現(xiàn)方法示例
這篇文章主要介紹了JavaScript選擇排序算法原理與實(shí)現(xiàn)方法,簡(jiǎn)單分析了選擇排序算法的概念、原理并結(jié)合實(shí)例形式分析了JavaScript選擇排序算法的相關(guān)實(shí)現(xiàn)技巧與操作注意事項(xiàng),需要的朋友可以參考下2018-08-08IE9+已經(jīng)不對(duì)document.createElement向下兼容的解決方法
這篇文章主要介紹了IE9+已經(jīng)不對(duì)document.createElement向下兼容的解決方法,需要的朋友可以參考下2015-09-09IE事件對(duì)象(The Internet Explorer Event Object)
不同于DOM事件對(duì)象,基于Event Handler授權(quán)這種方式,IE事件對(duì)象可以用不同的方式進(jìn)行訪問。當(dāng)一個(gè)事件Handler通過DOM 0 級(jí)的方式被授權(quán),則這個(gè)事件對(duì)象將作為window對(duì)象的屬性而存在2012-06-06javascript實(shí)現(xiàn)Table排序的方法
這篇文章主要介紹了javascript實(shí)現(xiàn)Table排序的方法,涉及javascript針對(duì)表格對(duì)象的獲取與排序相關(guān)技巧,需要的朋友可以參考下2015-05-05js實(shí)現(xiàn)簡(jiǎn)單分頁(yè)導(dǎo)航欄效果
這篇文章主要為大家詳細(xì)介紹了js實(shí)現(xiàn)簡(jiǎn)單分頁(yè)導(dǎo)航欄效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-06-06