什麼是閉包?看看閉包有哪些作用?以下本篇文章帶大家聊聊javascript閉包。有一定的參考價值,有需要的朋友可以參考一下,希望對大家有幫助。
在前端學習的過程中,我們難免會遇到許許多多的問題,那麼今天我們來以一個初學者的角度來談談兩個問題:
什麼是閉包?
閉包有哪些作用?
其實我們在學習javascript的時候閉包無所不在,你只需要能夠辨識並接受它。閉包並不是一個需要學習的新的語法或模式才能使用的工具,閉包是基於詞法作用域書寫程式碼時所產生的自然結果。我們幾乎不用在寫程式碼時刻意去創建閉包。
相信此時已經有不少小夥伴心裡在嘀咕,這詞法作用域有是個啥,不用慌,且聽我緩緩道來,簡而言之,詞法作用域就是定義在詞法階段的作用域。換句話說,詞法作用域是由你在寫程式碼時將變數和區塊級作用域寫在哪裡決定的,因此當詞法分析處理器處理程式碼時會保持作用域不變(大部分情況是這樣的) 。 ——《你不知道的javascript上卷》
先來個例子
function test(){ var arr = [] for(var i=0;i<10;i++){ arr[i]=function(){ console.log(i); } } return arr } var myArr = test() // myArr[0]() // myArr[1]() // ... for(var j = 0; j < 10; j++){ myArr[j]() } //为了避免繁琐此处使用了第二个循环来调用test函数里第一个循环里函数的打印出十个结果
我們先來分析一下這段程式碼: 當這段程式碼執行時,依照常理分析應該是依序印出0~9,十個數字;但是for迴圈運行時不需要時間(以微秒計算忽略不計),當函數test,return arr時,arr[ ]裡面是10個function(){console.log(i);},此時陣列裡的函數沒有執行,當var myArr = test()呼叫test函數時,由於for迴圈執行的時間忽略不計,所以此時i已經是10,所以印出的是10個10。
相信此時就會有人問,這與我們要講的閉包又有什麼關係呢,那如果我們把這段程式碼稍微修改一下,改變成一個累加器,要如何實現他呢?
相信此時會有大佬表示那還不簡單嗎?
把var定義改成let定義,使得第一個for迴圈成為一個區塊級作用域,那麼就可以變成累加器了。當然是沒有問題的,
但我們今天講的是在ES5的時候如何實作累加器。那我們再看看下面這段程式碼:
function test(){ var arr = [] for(var i=0;i<10;i++){ (function(j){ arr[j]=function(){ console.log(j); } })(i) } return arr } var myArr = test() for(var j = 0; j < 10; j++){ myArr[j]() }
細心地朋友一定會發現,這不就是把在循環裡面的函數體改成一個自執行函數嘛,但是此時輸出的結果就是 從0~9依序輸出十個數字,而這裡面就包含了閉包,當我們開始執行這段程式碼的時候第二個for迴圈會呼叫十次,當每個自執行函數執行時會建立一個自執行函數的AO對象,這個自執行函數的AO對象裡就存在一個屬性名為j,照常理而言自執行函數執行完畢之後,它的AO對象就應該被銷毀了,但是當myarr[j] ( )執行時,現在作用域鏈頂端的arr[j]的AO對象裡找屬性名j,但是沒有找到,順著作用域鏈往下找,在自執行函數的AO對象裡找到了,所以當自執行函數結束時,它的AO物件並不會被垃圾回收機制回收,否則當myarr[j] ()執行時就會報錯,此時就形成了閉包。
#我們再來舉個例子
function a(){ function b(){ var bbb = 234 console.log(aaa); } var aaa = 123 return b // b出生在a里面,但是被保存出去了 } var glob = 100 var demo = a() demo()
這段程式碼我們先用預編譯來分析,首先是定義一個全局的GO對象,在找全局的聲明找全局變量聲明,將變量聲明作為GO 的屬性名,值為undefined,在全局找函數聲明,將函數名作為GO 物件的屬性名,值賦予函數體。此時應為GO{ glob: undefined--->100 ; demo: undefined ; a: fa(){} }; 再對函數a創建一個AO{ aaa:undefined--->123;b: fb(){} },最後再對函數a裡面的函數b進行預編譯創建一個b的AO{ b: undefined-- ->234};此時作用域鏈的順序為1. 函數b的AO物件;2. 函數a的AO物件;3. 全域GO物件。當我們印函數b裡面的aaa時,先從作用域鏈的頂端開始找,在函數b的AO物件裡沒有aaa,那就會順著作用域鏈往下找,找到第二層函數a的AO物件是找到了aaa的值為123,輸出結果。
如果我们没有从预编译的角度去分析就会认为此时的aaa应该会报错的,当var demo = a()执行时,当a函数执行结束,那么a对应的AO对象应该被销毁了,照常理分析当我们执行demo时作用域链此时应该会创建b的AO对象和GO对象,此时只有b的AO对象,没有a的AO对象,应该不能打印出aaa的值,但是此时aaa的值为123,则说明a的AO对象没有被销毁,那么为什么呢?原因就在于这里创建了闭包,当var demo = a()执行结束之后,垃圾回收机制会来问,a函数老兄,我看你都执行完毕了,你的运行内存是不是可以给我释放了,但是此时a函数只能无奈摇摇头说道,老哥,我也不确定我有没有执行完毕,我执行完创建了一个b,但是b又不归我管,所以我也不确定b有没有被调用,所以我也不确定我有没有执行完毕,垃圾回收机制想了想,既然你都不知道那我就不回收了,要是回收了还没执行完的就该报错了,所以此时a的AO对象就没有被回收。
补充全面一点就是:当一个函数内部的函数被保存到函数外部时,就会产生闭包。
相信通过这两个例子,你已经对闭包有了一个大概的了解,那接下来我们说一下闭包有哪些作用。
闭包的作用
- 实现公有变量 例如:累加器(3.js)
- 做缓存
- 可以实现封装,属性私有化
- 模块化开发,防止污染全局变量
我们对闭包的作用也来一个例子(3.js)
var count = 0 function add() { return count++ } console.log(add()); console.log(add()); console.log(add());登入後複製这是一段比较普通的累加的代码,但是如果我们在实习甚至是工作的时,公司要求你把累加器封装成一个模块化的代码,那么
此时,为了模块化我们尽可能避免定义全局变量,但是不定义全局变量我们如何实现呢,此时我们就可以用到闭包了;function add() { var count = 0 function a() { ++count console.log(count); } return a } var res = add() res() res() //add函数结束之后,add的AO对象没有被销毁,因为add函数执行完了之后,返回的a不知道是否被调用就形成了闭包,这样 就能使得不使用全局变量也能封装成一个模块化的累加器。登入後複製结语
那么关于闭包以及闭包的作用相关的一些个人见解就是这些,目前对于闭包也只是一些浅显的了解,后期学习之后完善过后会出后续关于闭包的相关文章,感谢您的观看,欢迎批评斧正,一起共同进步。
【相关推荐:javascript视频教程、web前端】
以上是什麼是閉包?聊聊javascript中閉包,看看閉包有哪些作用?的詳細內容。更多資訊請關注PHP中文網其他相關文章!