成人怡红院-成人怡红院视频在线观看-成人影视大全-成人影院203nnxyz-美女毛片在线看-美女免费黄

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

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

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

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

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

場(chǎng)景思考

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

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

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

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

場(chǎng)景步驟:

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);   }; };
登錄后復(fù)制

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

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

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

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

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

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

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

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

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

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

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

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

我們繼續(xù)對(duì)比測(cè)試

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

// 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);   }; };
登錄后復(fù)制

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

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

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

閉包簡(jiǎn)介

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

閉包示例:

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

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

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

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

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

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

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

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

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

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

let a = '小馬哥'
登錄后復(fù)制

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

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

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

再定義一個(gè)函數(shù)

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

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

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

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

請(qǐng)謹(jǐn)記:函數(shù)的作用域鏈?zhǔn)窃趧?chuàng)建時(shí)就確定了,JS 引擎會(huì)創(chuàng)建函數(shù)時(shí),在該對(duì)象上添加一個(gè)名叫作用域鏈的屬性,該屬性包含著當(dāng)前函數(shù)的作用域以及父作用域,一直到全局作用域

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

垃圾回收機(jī)制淺析

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

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

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

上圖中,左下角的兩個(gè)值,沒(méi)有任何引用,所以可以釋放

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

判斷一個(gè)對(duì)象是否會(huì)被垃圾回收的標(biāo)準(zhǔn): 從全局對(duì)象 window 開(kāi)始,順著引用表能找到的都不是內(nèi)存垃圾,不會(huì)被回收掉。只有那些找不到的對(duì)象才是內(nèi)存垃圾,才會(huì)在適當(dāng)?shù)臅r(shí)機(jī)被 gc 回收

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

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

進(jìn)行斷點(diǎn)調(diào)試

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

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

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

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

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

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

當(dāng) info 在 debounce 函數(shù)內(nèi)部時(shí),進(jìn)行斷點(diǎn)調(diào)試

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

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

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

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

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

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

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

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

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

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; };
登錄后復(fù)制

2、自動(dòng)釋放(大多數(shù)的場(chǎng)景)

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

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);   }; };
登錄后復(fù)制

結(jié)論

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

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

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

推薦學(xué)習(xí):《JavaScript視頻教程》

贊(0)
分享到: 更多 (0)
?
網(wǎng)站地圖   滬ICP備18035694號(hào)-2    滬公網(wǎng)安備31011702889846號(hào)
狠狠躁夜夜躁青青草原| 好爽又高潮了毛片| 国模少妇无码一区二区三区| 精品成人乱色一区二区| 久久久久久妓女精品影院| 乱人伦人成品精国产在线| 免费A级毛片无码A∨蜜芽按摩| 欧美黑人又大又粗又长久久久| 人妻丰满熟妇A无码区| 成人无码AV网站在线观看| 高清性色生活片97| 国精品无码人妻一区二区三区| 久久久久黑人强伦姧人妻| 女邻居的大乳中文字幕理论| 日日摸夜夜添夜夜添AV| 性欧美ⅩXX1819内谢| 亚洲伊人久久综合成人网站| 99热久RE这里只有精品小草| 俄罗斯人和欧洲人的性格差异| 国产无遮挡裸露视频免费| 久久亚洲AV成人无码电影| 人妻无码一区二区视频| 欧美日韩精品一区二区在线播放 | 少妇一级无码精品| 亚洲精品国产第一综合99久久| 中文字幕人成乱码熟女免费| 在办公室被C到呻吟的动态图 | 国产成人亚洲综合| 久久精品国产一区二区三区不卡| 内射人妻无码色AV无码| 天堂8中文在线最新版官网| 日韩精品无码免费专区网站 | 久久精品国产亚洲AV果冻传媒| 女人被躁到高潮嗷嗷叫小说百度| 熟妇的奶头又大又粗视频| 亚洲男同帅GAY片在线观看| JAGNEXSMAX在日本| 国产亚洲欧美日韩在线一区二区三 | 久久精品国产99精品亚洲蜜桃| 欧洲Av无码放荡人妇网站| 西西人体444WWW大胆无码视 | 免费女同毛片在线播放| 天堂А√在线最新版在线8| 色偷偷人人澡久久超碰97| 亚洲国产精品嫩草影院久久 | 人人妻人人澡人人乐DVD| 亚洲成a人蜜臀AV在线播放| AV夜夜躁狠狠躁日日躁| 国精一二二产品无人区免费应用| 女人与公拘交酡ZOZO| 亚洲AV成人精品网站在线播放| 91精品人妻一区二区三在线| 国产欧美久久久久久精品一区二区| 母亲とが话していま在线观看| 无码人妻久久久一区二区三区 | 女人的选择HD中字| 性生交大片免费看| 亚洲AV日韩AV综合AⅤXXX| 337P人体粉嫩胞高清大图AV| JIZZYOU中国少妇高潮| 豆国产96在线 | 亚洲| 久久久久久精品免费免费自慰 | 亚洲人成网站18禁止大| 吃奶呻吟打开双腿做受视频| 粉嫩av.con| 没带罩子让他吃了一天药会怎样| 无码免费伦费影视在线观看 | 国产精品亚洲专区无码唯爱网| 女技师强制高潮18XXXX按摩| 亚洲AV永久无码精品网站| 成 人影片免费观看| 久久亚洲AV成人无码电影| 小受呻吟高潮GV在线观看| YYYY11111少妇影院| 精品久久久无码人妻中文字幕| 色欲香天天天综合网站小说| 中国女人内谢69XXXX免费视| 国产性猛交╳XXX乱大交| 人妻少妇性色精品专区av| 一边下奶一边吃面膜视频讲解| 国产免费无遮挡吸乳视频在线观看 | 久久久久亚洲精品无码系列| 女自慰喷水免费观看WWW久久| 亚洲爆乳无码专区WWW| 国产AV鲁丝一区二区二区| 奶头大他一口都含不住| 亚洲日韩精品A∨片无码加勒比| 国产浮力第一页草草影院| 热爆料-热门吃瓜-黑料不打烊| 一本色道久久88亚洲精品综合| 国精产品一区二区三区糖心| 熟女精品视频一区二区三区| WWWXXX国产| 男人GAY自慰吞精网站| 亚洲熟悉妇女XXX妇女AV| 国产强奷在线播放| 天天躁日日躁狠狠躁AV| 菠萝菠萝蜜菠萝菠萝5| 欧美成人精品激情在线观看| 亚洲孕妇精品无码av| 和丰满少妇作爱过程视频| 无码人妻精品一区二| 大陆熟妇丰满多毛XXXX| 人妻AV无码专区| 97久久精品亚洲中文字幕无码| 久久亚洲AV成人无码国产| 亚洲线精品一区二区三区影音先锋| 国产午夜片无码区在线观看爱情网| 天天躁狠狠躁狠狠躁性色AV| 芳芳好紧好滑好湿好爽| 日本高清视频网站WWW| TOUGHTESTED硬汉系列| 欧美黑人又粗又大高潮喷水| 中文字幕人妻偷伦在线视频| 久久婷婷综合色丁香五月| 少妇又紧又深又湿又爽视频| 波多野结衣 美乳人妻| 强奷秘书吸乳免费观看| 亚洲成无码电影在线观看| 国产在线精品一区二区高清不卡| 无码高清一区二区三区| 国产成人A在线观看视频免费| 色又黄又爽18禁免费网站现观看| 成人免费无码不卡毛片视频| 日本精品无码一区二区三区久久久| JAPANESE精品中国少妇| 琪琪女色窝窝777777| おやすみせくよ晚安免费影院| 欧美性色黄大片WWW喷水| А√最新版在线天堂| 人妻丰满熟妇av无码久久洗澡| WWW插插插无码免费视频网站| 欧洲一卡2卡三卡4卡 乱码 | 岳好紧好紧我要进去了电影| 麻豆XXXXXX在线观看| 中文字幕在线不卡一区二区 | 成人免费无码H黄网站WWW| 日韩成人无码中文字幕| 国产AV无码国产AV毛片| 调教小SAO货撅起打屁股作文| 国产精品视频一区二区三区不卡 | 久久国产精品成人免费| 又爽又黄又无遮挡的视频在线观看 | 欧美 日韩 国产 另类 图片区 | 精品人伦一区二区三区蜜桃| 亚洲色偷偷偷综合网 | 自拍偷区亚洲综合美利坚| 欧美成性视頻XXXⅩXXX| 被公疯狂玩弄的漂亮人妻| 色婷婷综合久久久中文字幕 | 爆乳女教师 高清BD| 舌尖伸入湿嫩蜜汁呻吟| 国产三级久久久精品麻豆三级| 亚洲国产成人爱AV网站| 看成年女人午夜毛片免费| AV无码中出一区二区三区| 日韩精品无码一区二区视频 | 免费看奶头视频的网站| JAPANESE丰满少妇最高潮| 色一乱一伦一图一区二区精品| 国产乱妇无码大片在线观看| 亚洲精品一卡2卡三卡4卡5卡| 麻豆国产果冻自制传媒| а√在线官网在线| 无码高潮喷吹在线播放亚洲| 精品国产一区二区三区吸毒| 真实的国产乱XXXX在线| 日本XXX色视频| HEYZO无码综合国产精品| 少妇 黑人 欧美 亚洲| 好男人WWW在线观看视频| 中文字幕AV无码免费一区| 老头的老枪又粗又大| 啊!摁摁~啊!用力~快点视频| 无码国产精成人午夜视频不卡| 精品乱人伦一区二区三区| 99国产精品99久久久久久| 爽到憋不住潮喷大喷水视频| 含羞草实验室隐藏路径2023| 在线视频一区二区三区| 揉大N呻吟水多大棒子| 狠狠综合久久狠狠88亚洲| 666西方最大但人文艺术| 天美传媒MV免费观看完整| 久久精品99国产精品蜜桃| 啊灬啊灬啊灬快灬高潮了I | 国产毛片久久久久久国产毛片| 夜夜揉揉日日人人| 色既是空在线观看| 精品久久久久久无码免费| wwwxxx一区二区| 亚洲AV福利天堂一区二区三| 妺妺晚上扒我内裤吃我精子| 国产AV无码专区亚洲AV琪琪| 亚洲中文字幕久久无码精品| 日本三线和韩国三线品牌对比| 护士HD老师FREE性ⅩⅩⅩⅩ| W永久939W乳液78| 亚洲AV永久无码精品秋霞电影影院| 欧美熟妇精品一区二区三区|