日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

环境与作用域与闭包

發(fā)布時間:2024/1/8 编程问答 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 环境与作用域与闭包 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

環(huán)境與作用域

在瀏覽器環(huán)境下,全局?jǐn)?shù)據(jù)默認(rèn)不會被回收,直到瀏覽器關(guān)閉

全局?jǐn)?shù)據(jù)可以滲透到函數(shù)內(nèi)部,被函數(shù)內(nèi)部訪問

函數(shù)的環(huán)境與作用域

  • 函數(shù)體內(nèi)定義的數(shù)據(jù)默認(rèn)在函數(shù)內(nèi)部使用

    function exam() {var a = 1let b = 2const c = 3console.log(a) // 1console.log(b) // 2console.log(c) // 3 } exam() console.log(a, b, c) // Uncaught ReferenceError: a is not defined
  • 子級函數(shù)內(nèi)定義數(shù)據(jù)不會向父級傳遞,但可以訪問父級數(shù)據(jù)

    function exam() {var a = 1function test() {let b = 2console.log(a) // 1}test()console.log(b) // Uncaught ReferenceError: b is not defined at exam } exam()
  • 函數(shù)不調(diào)用不開辟空間

  • 函數(shù)調(diào)用一次,開辟一次空間,每次都不相同

  • 函數(shù)運(yùn)行結(jié)束,內(nèi)部數(shù)據(jù)未被使用則釋放空間

    function exam() {let n = 1console.log(++n) } exam() // 2 exam() // 2 exam() // 2
  • 若函數(shù)內(nèi)部存在數(shù)據(jù)被使用則不釋放空間

    function test() {let n = 1return function () {console.log(++n)} } let run = test() run() // 2 run() // 3 run() // 4 run() // 5

構(gòu)造函數(shù)中作用域形態(tài)

  • new會開辟空間存放數(shù)據(jù),在實(shí)例銷毀之前,對實(shí)例數(shù)據(jù)的更改都會反映在構(gòu)建的內(nèi)存空間中

    function Fn() {let n = 1this.sum = function () {console.log(++n)} } const a = new Fn() a.sum() // 2 a.sum() // 3 a.sum() // 4 a.sum() // 5

塊級作用域

es6中定義的let、const為塊級作用域

{var a = 3console.log(a) // 3 } {let b = 2console.log(b) // 2 } console.log(a) // 3 console.log(b) // ReferenceError: b is not defined

for中的var、let

  • for 自帶塊級特性

    for (var i = 0; i < 2; i++) {console.log(i) // 0 1 } for (let j = 0; j < 2; j++) {console.log(j) // 0 1 } console.log(i) // 2 console.log(j) // j is not defined
  • 由于var不存在塊級作用域,在存在定時器等特殊狀況的情況下得不到預(yù)期結(jié)果

    for (var i = 0; i < 2; i++) {setTimeout(function() {console.log(i) // 2 2}, 1000); } for (let j = 0; j < 2; j++) {setTimeout(function() {console.log(j) // 0 1}, 1000); } console.log(i) // 2

    var表示在全局定義數(shù)據(jù)i,在定時器函數(shù)構(gòu)建的空間中不會存儲變量i而是從父級作用域中取值

    for作為同步代碼先行執(zhí)行后i=2,等定時器函數(shù)運(yùn)行時i已經(jīng)被賦值為2

    let具備塊作用域,定時函數(shù)在構(gòu)建空間存儲時會將當(dāng)前i的值保存在自己的作用域空間中,使用時先從自身作用域空間查找使用

模擬出var的塊級作用域

for (var i = 0; i <= 3; i++) {(function (i) {setTimeout(() => {console.log(i) // 0 1 2 3}, 1000)})(i) } console.log('window.i',i) // 4

利用自調(diào)用函數(shù)將循環(huán)中的i作為實(shí)參傳遞(實(shí)際是建立了循環(huán)次數(shù)**(4次)**的新函數(shù)作用域空間)

定時器函數(shù)構(gòu)建作用域空間時會攜帶每次傳遞來的實(shí)參作為自身參數(shù)

閉包

函數(shù)可以訪問其他函數(shù)作用域內(nèi)的數(shù)據(jù)

閉包運(yùn)用之?dāng)?shù)組區(qū)間

let arr = [1,2,3,45,66,5,63,345,4251,3] function between(start,end) {return function (value) {return value>= start && value<=end} } console.log(arr.filter(between(5,100))) // [45, 66, 5, 63]

filter需要一個回調(diào)函數(shù)

建立函數(shù)between返回filter需要的回調(diào)函數(shù)

由于作用域問題,在回調(diào)函數(shù)執(zhí)行時獲取父級作用域空間的數(shù)據(jù)的(即形成閉包)

因此傳遞實(shí)參給between以達(dá)到控制數(shù)組區(qū)間的目的

閉包的內(nèi)存泄漏

<body><button desc="storage">內(nèi)存</button><button desc="leak">泄漏</button><button desc="problem">問題</button><script>let btns = document.querySelectorAll('button')btns.forEach(item=>{let desc = item.getAttribute('desc')item.addEventListener('click',()=>{console.log('desc:',desc)console.log('item:',item)})item = null})</script> </body>

由于函數(shù)內(nèi)數(shù)據(jù)使用時不釋放內(nèi)存的原因,對于事件處理函數(shù)中會存儲DOM對象本身,對于只需要獲取特定數(shù)據(jù)的操作來說,無疑導(dǎo)致了大量的內(nèi)存浪費(fèi),過多的事件處理函數(shù)可能導(dǎo)致內(nèi)存泄漏問題

因此在事件處理函數(shù)之外獲取數(shù)據(jù),通過閉包可以在事件處理函數(shù)中獲取數(shù)據(jù),之后在手動釋放內(nèi)存空間,以達(dá)到優(yōu)化程序的作用

總結(jié)

以上是生活随笔為你收集整理的环境与作用域与闭包的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。