gmnon.cn-疯狂蹂躏欧美一区二区精品,欧美精品久久久久a,高清在线视频日韩欧美,日韩免费av一区二区

站長資訊網(wǎng)
最全最豐富的資訊網(wǎng)站

一文深析閉包用多是否會造成內(nèi)存泄露

閉包,是JS中的一大難點;網(wǎng)上有很多關(guān)于閉包會造成內(nèi)存泄露的描述,說閉包會使其中的變量的值始終保持在內(nèi)存中,一般都不太推薦使用閉包

而項目中確實有很多使用閉包的場景,比如函數(shù)的節(jié)流與防抖

那么閉包用多了,會造成內(nèi)存泄露嗎?

場景思考

以下案例: A 頁面引入了一個 debounce 防抖函數(shù),跳轉(zhuǎn)到 B 頁面后,該防抖函數(shù)中閉包所占的內(nèi)存會被 gc 回收嗎?

該案例中,通過變異版的防抖函數(shù)來演示閉包的內(nèi)存回收,此函數(shù)中引用了一個內(nèi)存很大的對象 info(42M的內(nèi)存),便于明顯地對比內(nèi)存的前后變化

注:可以使用 Chrome 的 Memory 工具查看頁面的內(nèi)存大小:

一文深析閉包用多是否會造成內(nèi)存泄露

場景步驟:

1) util.js 中定義了 debounce 防抖函數(shù)

// util.js`let info = {  arr: new Array(10 * 1024 * 1024).fill(1),  timer: null};export const debounce = (fn, time) => {  return function (...args) {     info.timer && clearTimeout(info.timer);     info.timer = setTimeout(() => {       fn.apply(this, args);     }, time);   }; };
登錄后復制

2) A 頁面中引入并使用該防抖函數(shù)

import { debounce } from './util';mounted() {    this.debounceFn = debounce(() => {      console.log('1');     }, 1000) }
登錄后復制

  • 抓取 A 頁面內(nèi)存: 57.1M

一文深析閉包用多是否會造成內(nèi)存泄露

3) 從 A 頁面跳轉(zhuǎn)到 B 頁面,B 頁面中沒有引入該 debounce 函數(shù)

問題: 從 A 跳轉(zhuǎn)到 B 后,該函數(shù)所占的內(nèi)存會被釋放掉嗎?

  • 此時,抓取 B 頁面內(nèi)存: 58.1M

一文深析閉包用多是否會造成內(nèi)存泄露

  • 刷新 B 頁面,該頁面的原始內(nèi)存為: 16.1M

一文深析閉包用多是否會造成內(nèi)存泄露

結(jié)論: 前后對比發(fā)現(xiàn),從 A 跳轉(zhuǎn)到 B 后,B 頁面內(nèi)存增大了 42M,證明該防抖函數(shù)所占的內(nèi)存沒有被釋放掉,即造成了內(nèi)存泄露

為什么會這樣呢? 按理說跳轉(zhuǎn) B 頁面后,A 頁面的組件都被銷毀掉了,那么 A 頁面所占的內(nèi)存應該都被釋放掉了啊?

我們繼續(xù)對比測試

4) 如果把 info 對象放到 debounce 函數(shù)內(nèi)部,從 A 跳轉(zhuǎn)到 B 后,該防抖函數(shù)所占的內(nèi)存會被釋放掉嗎?

// util.js`export const debounce = (fn, time) => { let info = {  arr: new Array(10 * 1024 * 1024).fill(1),  timer: null  };  return function (...args) {     info.timer && clearTimeout(info.timer);     info.timer = setTimeout(() => {       fn.apply(this, args);     }, time);   }; };
登錄后復制

按照步驟 4 的操作,重新從 A 跳轉(zhuǎn)到 B 后,B 頁面抓取內(nèi)存為16.1M,證明該函數(shù)所占的內(nèi)存被釋放掉了

為什么只是改變了 info 的位置,會引起內(nèi)存的前后變化?

要搞懂這個問題,需要理解閉包的內(nèi)存回收機制

閉包簡介

閉包:一個函數(shù)內(nèi)部有外部變量的引用,比如函數(shù)嵌套函數(shù)時,內(nèi)層函數(shù)引用了外層函數(shù)作用域下的變量,就形成了閉包。最常見的場景為:函數(shù)作為一個函數(shù)的參數(shù),或函數(shù)作為一個函數(shù)的返回值時

閉包示例:

function fn() {   let num = 1;   return function f1() {     console.log(num);   };} let a = fn();a();
登錄后復制

上面代碼中,a 引用了 fn 函數(shù)返回的 f1 函數(shù),f1 函數(shù)中引入了內(nèi)部變量 num,導致變量 num 滯留在內(nèi)存中

打斷點調(diào)試一下

一文深析閉包用多是否會造成內(nèi)存泄露

展開函數(shù) f 的 Scope(作用域的意思)選項,會發(fā)現(xiàn)有 Local 局部作用域、Closure 閉包、Global 全局作用域等值,展開 Closure,會發(fā)現(xiàn)該閉包被訪問的變量是 num,包含 num 的函數(shù)為 fn

總結(jié)來說,函數(shù) f 的作用域中,訪問到了fn 函數(shù)中的 num 這個局部變量,從而形成了閉包

所以,如果真正理解好閉包,需要先了解閉包的內(nèi)存引用,并且要先搞明白這幾個知識點:

  • 函數(shù)作用域鏈
  • 執(zhí)行上下文
  • 變量對象、活動對象

函數(shù)的內(nèi)存表示

先從最簡單的代碼入手,看下變量是如何在內(nèi)存中定義的

let a = '小馬哥'
登錄后復制

這樣一段代碼,在內(nèi)存里表示如下

一文深析閉包用多是否會造成內(nèi)存泄露

在全局環(huán)境 window 下,定義了一個變量 a,并給 a 賦值了一個字符串,箭頭表示引用

再定義一個函數(shù)

let a = '小馬哥'function fn() {  let num = 1}
登錄后復制

內(nèi)存結(jié)構(gòu)如下:

一文深析閉包用多是否會造成內(nèi)存泄露

特別注意的是,fn 函數(shù)中有一個 [[scopes]] 屬性,表示該函數(shù)的作用域鏈,該函數(shù)作用域指向全局作用域(瀏覽器環(huán)境就是 window),函數(shù)的作用域是理解閉包的關(guān)鍵點之一

請謹記:函數(shù)的作用域鏈是在創(chuàng)建時就確定了,JS 引擎會創(chuàng)建函數(shù)時,在該對象上添加一個名叫作用域鏈的屬性,該屬性包含著當前函數(shù)的作用域以及父作用域,一直到全局作用域

函數(shù)在執(zhí)行時,JS 引擎會創(chuàng)建執(zhí)行上下文,該執(zhí)行上下文會包含函數(shù)的作用域鏈(上圖中紅色的線),其次包含函數(shù)內(nèi)部定義的變量、參數(shù)等。在執(zhí)行時,會首先查找當前作用域下的變量,如果找不到,就會沿著作用域鏈中查找,一直到全局作用域

垃圾回收機制淺析

現(xiàn)在各大瀏覽器通常用采用的垃圾回收有兩種方法:標記清除、引用計數(shù)

這里重點介紹 "引用計數(shù)"(reference counting),JS 引擎有一張"引用表",保存了內(nèi)存里面所有的資源(通常是各種值)的引用次數(shù)。如果一個值的引用次數(shù)是0,就表示這個值不再用到了,因此可以將這塊內(nèi)存釋放

一文深析閉包用多是否會造成內(nèi)存泄露

上圖中,左下角的兩個值,沒有任何引用,所以可以釋放

如果一個值不再需要了,引用數(shù)卻不為0,垃圾回收機制無法釋放這塊內(nèi)存,從而導致內(nèi)存泄漏

判斷一個對象是否會被垃圾回收的標準: 從全局對象 window 開始,順著引用表能找到的都不是內(nèi)存垃圾,不會被回收掉。只有那些找不到的對象才是內(nèi)存垃圾,才會在適當?shù)臅r機被 gc 回收

分析內(nèi)存泄露的原因

回到最開始的場景,當 info 在 debounce 函數(shù)外部時,為什么會造成內(nèi)存泄露?

進行斷點調(diào)試

一文深析閉包用多是否會造成內(nèi)存泄露

展開 debounce 函數(shù)的 Scope選項,發(fā)現(xiàn)有兩個 Closure 閉包對象,第一個 Closure 中包含了 info 對象,第二個 Closure 閉包對象,屬于 util.js 這個模塊

內(nèi)存結(jié)構(gòu)如下:

一文深析閉包用多是否會造成內(nèi)存泄露

當從 A 頁面切換到 B 頁面時,A 頁面被銷毀,只是銷毀了 debounce 函數(shù)當前的作用域,但是 util.js 這個模塊的閉包卻沒有被銷毀,從 window 對象上沿著引用表依然可以查找到 info 對象,最終造成了內(nèi)存泄露

一文深析閉包用多是否會造成內(nèi)存泄露

當 info 在 debounce 函數(shù)內(nèi)部時,進行斷點調(diào)試

一文深析閉包用多是否會造成內(nèi)存泄露

其內(nèi)存結(jié)構(gòu)如下:

一文深析閉包用多是否會造成內(nèi)存泄露

當從 A 頁面切換到 B 頁面時,A 頁面被銷毀,同時會銷毀 debounce 函數(shù)當前的作用域,從 window 對象上沿著引用表查找不到 info 對象,info 對象會被 gc 回收

一文深析閉包用多是否會造成內(nèi)存泄露

閉包內(nèi)存的釋放方式

1、手動釋放(需要避免的情況)

如果將閉包引用的變量定義在模塊中,這種會造成內(nèi)存泄露,需要手動釋放,如下所示,其他模塊需要調(diào)用 clearInfo 方法,來釋放 info 對象

可以說這種閉包的寫法是錯誤的 (不推薦), 因為開發(fā)者需要非常小心,否則稍有不慎就會造成內(nèi)存泄露,我們總是希望可以通過 gc 自動回收,避免人為干涉

let info = {  arr: new Array(10 * 1024 * 1024).fill(1),  timer: null};export const debounce = (fn, time) => {  return function (...args) {     info.timer && clearTimeout(info.timer);     info.timer = setTimeout(() => {       fn.apply(this, args);     }, time);   }; };export const clearInfo = () => {   info = null; };
登錄后復制

2、自動釋放(大多數(shù)的場景)

閉包引用的變量定義在函數(shù)中,這樣隨著外部引用的銷毀,該閉包就會被 gc 自動回收 (推薦),無需人工干涉

export const debounce = (fn, time) => {  let info = {    arr: new Array(10 * 1024 * 1024).fill(1),    timer: null   };  return function (...args) {     info.timer && clearTimeout(info.timer);     info.timer = setTimeout(() => {       fn.apply(this, args);     }, time);   }; };
登錄后復制

結(jié)論

綜上所述,項目中大量使用閉包,并不會造成內(nèi)存泄漏,除非是錯誤的寫法

絕大多數(shù)情況,只要引用閉包的函數(shù)被正常銷毀,閉包所占的內(nèi)存都會被 gc 自動回收。特別是現(xiàn)在 SPA 項目的盛行,用戶在切換頁面時,老頁面的組件會被框架自動清理,所以我們可以放心大膽的使用閉包,無需多慮

理解了閉包的內(nèi)存回收機制,才算徹底搞懂了閉包。以上關(guān)于閉包的理解,如果有誤,歡迎指正

推薦學習:《JavaScript視頻教程》

贊(0)
分享到: 更多 (0)
?
網(wǎng)站地圖   滬ICP備18035694號-2    滬公網(wǎng)安備31011702889846號
gmnon.cn-疯狂蹂躏欧美一区二区精品,欧美精品久久久久a,高清在线视频日韩欧美,日韩免费av一区二区
黄色激情在线视频| 91福利免费观看| 三级黄色片免费观看| 国产毛片视频网站| 天天干天天色天天干| 国模杨依粉嫩蝴蝶150p| 男人添女人下部视频免费| 自拍偷拍21p| 激情视频综合网| 国产91在线视频观看| 国产欧美精品aaaaaa片| 三级黄色片免费观看| 亚欧在线免费观看| 91av在线免费播放| 欧美日韩精品在线一区二区| 2025韩国大尺度电影| 日韩av卡一卡二| 亚洲综合日韩欧美| 国产免费999| 在线免费视频a| 日批视频在线免费看| 国产深夜男女无套内射| 免费一级特黄特色毛片久久看| 亚洲天堂一区二区在线观看| 国产美女18xxxx免费视频| 国产wwwxx| 黄色永久免费网站| 久久国产精品国产精品| 天天爽天天爽夜夜爽| 国产一伦一伦一伦| 亚洲18在线看污www麻豆| 日韩av播放器| 国产精品自拍视频在线| 亚洲网中文字幕| 看一级黄色录像| 999一区二区三区| 亚洲人精品午夜射精日韩| 缅甸午夜性猛交xxxx| www国产精品内射老熟女| 黄页免费在线观看视频| 免费高清在线观看免费| 一级黄色特级片| 水蜜桃在线免费观看| 美女扒开大腿让男人桶| 日本成年人网址| 国产精品久久久久久9999| 色一情一乱一乱一区91| 女人天堂av手机在线| 色诱视频在线观看| 992tv人人草| 亚洲熟妇无码一区二区三区| 日本精品久久久久中文字幕| 亚洲第一区第二区第三区| 久久久久久久久网| 北条麻妃av高潮尖叫在线观看| 四季av一区二区三区| www.男人天堂网| 91激情视频在线| 日韩精品久久一区二区| 激情网站五月天| 99中文字幕在线观看| 日韩免费毛片视频| 无码人妻aⅴ一区二区三区日本| 国产特级黄色大片| 免费不卡av网站| 国语对白做受xxxxx在线中国| 亚洲国产精品影视| 88av.com| 久久久久免费看黄a片app| 99re6在线观看| 91黄色小网站| 99亚洲国产精品| 911福利视频| 91猫先生在线| 久久这里只有精品8| 中文字幕22页| 日韩精品一区中文字幕| av动漫在线播放| 涩涩网站在线看| 国产精品视频黄色| 无码人妻h动漫| 99久久国产综合精品五月天喷水| 亚洲免费在线播放视频| 91网址在线播放| 国产三级日本三级在线播放| 国产妇女馒头高清泬20p多| 国产高潮呻吟久久久| 亚洲成人天堂网| 亚洲精品一二三四五区| 欧美色图另类小说| 国产 日韩 欧美在线| 麻豆视频传媒入口| 免费久久久久久| 一级黄色高清视频| 日本高清免费在线视频| 性生生活大片免费看视频| 亚洲色图 在线视频| www日韩在线观看| 伊人影院综合在线| 国产成人美女视频| 香蕉视频在线网址| 一本色道久久88亚洲精品综合| 色婷婷综合在线观看| 亚洲在线观看网站| 亚洲黄色网址在线观看| 女人床在线观看| 欧美成人三级在线视频| 霍思燕三级露全乳照| 欧美极品欧美精品欧美| www..com日韩| 四虎永久在线精品无码视频| 天天干在线影院| 亚洲精品视频三区| 樱空桃在线播放| 精品无码国模私拍视频| 浮妇高潮喷白浆视频| 88av.com| 日韩视频 中文字幕| 黄色一级在线视频| 99视频在线视频| 大地资源第二页在线观看高清版| www.偷拍.com| aa在线免费观看| 97人人爽人人| 免费看又黄又无码的网站| 99精品视频播放| 久久久一二三四| 精品视频免费在线播放| www.夜夜爽| www.99热这里只有精品| 亚洲 激情 在线| 青青青在线视频播放| the porn av| 波多野结衣av一区二区全免费观看| 日韩精品一区二区三区久久| www.国产福利| 麻豆av免费在线| 日韩a级在线观看| 亚洲77777| 青青草国产精品视频| 久久久久xxxx| 成人免费xxxxx在线视频| www.色.com| 国产精品人人妻人人爽人人牛| www亚洲国产| 五月婷婷激情久久| 免费在线a视频| 欧美 日韩 国产精品| 亚洲第一天堂久久| 成人黄色一区二区| 日韩在线综合网| 无码人妻精品一区二区蜜桃网站| 中文字幕免费高清在线| a在线视频观看| 免费网站永久免费观看| 日本免费在线视频观看| 欧美亚洲日本在线观看| 日本免费黄视频| 黄色成人在线看| 男女激情免费视频| 日本人妻伦在线中文字幕| 性久久久久久久久久久久久久| 亚洲老女人av| 亚洲精品www.| 天天插天天操天天射| 999香蕉视频| 欧美一级裸体视频| 在线免费视频a| 亚洲五月天综合| 日本xxxx黄色| 亚洲美女性囗交| √天堂资源在线| 蜜臀av.com| 99久久久精品视频| 国产真人做爰毛片视频直播| 超碰成人免费在线| 欧美人成在线观看| 国产精品久久..4399| 男人的天堂狠狠干| 亚洲精品中文字幕无码蜜桃| 妺妺窝人体色www在线小说| 一本色道无码道dvd在线观看| 北条麻妃在线视频| 色片在线免费观看| 欧美 国产 精品| 男女私大尺度视频| 亚洲色精品三区二区一区| 狠狠干狠狠操视频| 300部国产真实乱| 免费 成 人 黄 色| 国产三级国产精品国产专区50| 久久成年人网站| 妺妺窝人体色777777| 免费观看成人网| 中文字幕av导航| 丰满爆乳一区二区三区| 亚洲激情在线观看视频| 日本一区二区免费高清视频| 成人网站免费观看入口| 在线免费观看视频黄|