久久精品五月,日韩不卡视频在线观看,国产精品videossex久久发布 ,久久av综合

站長資訊網
最全最豐富的資訊網站

一文深析閉包用多是否會造成內存泄露

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

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

那么閉包用多了,會造成內存泄露嗎?

場景思考

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

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

注:可以使用 Chrome 的 Memory 工具查看頁面的內存大小:

一文深析閉包用多是否會造成內存泄露

場景步驟:

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

// 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 頁面中引入并使用該防抖函數

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

  • 抓取 A 頁面內存: 57.1M

一文深析閉包用多是否會造成內存泄露

3) 從 A 頁面跳轉到 B 頁面,B 頁面中沒有引入該 debounce 函數

問題: 從 A 跳轉到 B 后,該函數所占的內存會被釋放掉嗎?

  • 此時,抓取 B 頁面內存: 58.1M

一文深析閉包用多是否會造成內存泄露

  • 刷新 B 頁面,該頁面的原始內存為: 16.1M

一文深析閉包用多是否會造成內存泄露

結論: 前后對比發現,從 A 跳轉到 B 后,B 頁面內存增大了 42M,證明該防抖函數所占的內存沒有被釋放掉,即造成了內存泄露

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

我們繼續對比測試

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

// 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 跳轉到 B 后,B 頁面抓取內存為16.1M,證明該函數所占的內存被釋放掉了

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

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

閉包簡介

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

閉包示例:

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

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

打斷點調試一下

一文深析閉包用多是否會造成內存泄露

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

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

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

  • 函數作用域鏈
  • 執行上下文
  • 變量對象、活動對象

函數的內存表示

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

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

這樣一段代碼,在內存里表示如下

一文深析閉包用多是否會造成內存泄露

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

再定義一個函數

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

內存結構如下:

一文深析閉包用多是否會造成內存泄露

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

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

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

垃圾回收機制淺析

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

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

一文深析閉包用多是否會造成內存泄露

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

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

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

分析內存泄露的原因

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

進行斷點調試

一文深析閉包用多是否會造成內存泄露

展開 debounce 函數的 Scope選項,發現有兩個 Closure 閉包對象,第一個 Closure 中包含了 info 對象,第二個 Closure 閉包對象,屬于 util.js 這個模塊

內存結構如下:

一文深析閉包用多是否會造成內存泄露

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

一文深析閉包用多是否會造成內存泄露

當 info 在 debounce 函數內部時,進行斷點調試

一文深析閉包用多是否會造成內存泄露

其內存結構如下:

一文深析閉包用多是否會造成內存泄露

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

一文深析閉包用多是否會造成內存泄露

閉包內存的釋放方式

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

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

可以說這種閉包的寫法是錯誤的 (不推薦), 因為開發者需要非常小心,否則稍有不慎就會造成內存泄露,我們總是希望可以通過 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、自動釋放(大多數的場景)

閉包引用的變量定義在函數中,這樣隨著外部引用的銷毀,該閉包就會被 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);   }; };
登錄后復制

結論

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

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

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

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

贊(0)
分享到: 更多 (0)
?
網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
久久精品五月,日韩不卡视频在线观看,国产精品videossex久久发布 ,久久av综合
不卡一区综合视频| 国产精品白丝一区二区三区| 精品国产美女a久久9999| 日本aⅴ精品一区二区三区 | 欧美国产小视频| 欧美成人精品午夜一区二区| 欧美亚洲三级| 国产精品porn| 国产高潮在线| 欧美+日本+国产+在线a∨观看| 国产一在线精品一区在线观看| 一区免费在线| 午夜性色一区二区三区免费视频| 日韩精品一区二区三区中文| 蜜桃tv一区二区三区| 欧美99久久| 另类av一区二区| 欧美日韩黄网站| 欧美韩日一区| 91九色精品国产一区二区| 模特精品在线| 日韩在线成人| 福利精品一区| 综合精品一区| 一区二区精品伦理...| 国产视频一区免费看| 伊人久久亚洲美女图片| 性色一区二区| 日本精品久久| 国产福利电影在线播放| 不卡一区2区| 四虎精品永久免费| 欧美国产免费| 欧美中文字幕一区二区| 在线免费观看亚洲| 久久精品国产999大香线蕉| 精品国产aⅴ| 亚洲天堂成人| 日韩中出av| 成人国产精品一区二区网站| 99成人在线| 久久精品天堂| 日韩视频二区| 美女尤物国产一区| 91成人在线网站| 国产欧美日韩一区二区三区在线| 亚洲一二三区视频| 国内精品伊人| 日韩影院免费视频| 国产成人a视频高清在线观看| 欧美日韩国产高清| 精品视频一区二区三区四区五区 | 国产亚洲精品久久久久婷婷瑜伽| 天堂va在线高清一区| 亚洲成人一区在线观看| 精品亚洲成人| 日韩三区免费| 国产精品嫩草99av在线| 极品裸体白嫩激情啪啪国产精品| 亚洲午夜黄色| 日韩精品欧美大片| 免费成人在线观看| 国产精品蜜芽在线观看| 欧美亚洲福利| 日韩成人亚洲| 中文字幕av一区二区三区人| 中文字幕在线视频网站| 午夜久久一区| 欧美精品99| 国产麻豆精品| 麻豆成全视频免费观看在线看| 日韩国产欧美三级| 国产成人黄色| 婷婷综合网站| 日韩精品专区| 国产乱子精品一区二区在线观看 | 日韩国产欧美在线视频| 在线亚洲免费| 欧美精品1区| 国产不卡精品| 最新亚洲激情| 日韩精品dvd| 日韩成人精品一区| 亚州精品视频| 色吊丝一区二区| 国产精品一区三区在线观看| 亚洲深夜影院| 91精品国产91久久久久久黑人| 综合激情一区| 国产精品综合色区在线观看| 久久一区精品| 日本亚洲不卡| 久久午夜视频| 亚洲小说欧美另类婷婷| 97精品在线| 麻豆精品99| 日韩精品社区| 色8久久久久| 视频一区视频二区中文字幕| 久久久久.com| 九九色在线视频| 国产成人在线中文字幕| 久久黄色影视| 欧美日韩一区二区三区不卡视频| 日本大胆欧美人术艺术动态| 夜夜嗨av一区二区三区网站四季av| 欧美一区二区三区激情视频| 成人久久一区| 色婷婷精品视频| 久久久夜夜夜| 国产综合精品一区| 国产精品99一区二区| 在线日韩欧美| 婷婷国产精品| 亚洲91精品| 久久精品不卡| 国产精品久久久久久久久久齐齐 | 亚洲一级大片| 久久九九电影| 蜜臀久久99精品久久久久宅男| 婷婷六月综合| 国产婷婷精品| 国产乱论精品| 中文字幕在线看片| 在线日韩av| 日本在线视频一区二区| 夜夜嗨一区二区| 久久夜色精品| 国产一区视频在线观看免费| 欧美黄色一区二区| 亚洲一区欧美二区| 影音国产精品| 日韩在线网址| 亚洲国产日韩欧美在线| 亚洲福利专区| 精品久久视频| 蜜臀av在线播放一区二区三区| 国产精品天堂蜜av在线播放| 久久的色偷偷| 国产精品视区| 男人的天堂亚洲一区| 午夜国产欧美理论在线播放| 美女久久一区| 日本精品一区二区三区在线观看视频| 日韩精品电影一区亚洲| 国产精品久久久久毛片大屁完整版 | 麻豆视频在线看| 久久国产中文字幕| 视频一区在线视频| 欧美天堂一区二区| 成人在线黄色| 一区视频在线| 日韩高清一区二区| 国产精品男女| 日本欧美不卡| 美女尤物久久精品| 国产日韩一区二区三区在线| 国产成人免费精品| 不卡中文一二三区| 欧美一区影院| 亚洲成人不卡| 视频一区二区国产| 成人va天堂| 国产欧美日本| 国产精品毛片久久| 好吊视频一区二区三区四区| 精品国产午夜肉伦伦影院| 日韩一区二区久久| 国产主播一区| 中文字幕日本一区二区| 国产亚洲欧美日韩在线观看一区二区 | 精品三级av| 99国内精品| 国产精久久久| 日韩天堂av| 国产欧美激情| 亚洲小说欧美另类婷婷| 亚洲一区二区av| 国内在线观看一区二区三区| 在线日韩欧美| 国产精品亚洲欧美日韩一区在线| 日韩欧美一区二区三区免费看| 免费在线观看视频一区| 精品中文在线| 99国产精品自拍| 麻豆视频观看网址久久| 亚洲少妇在线| 精品美女在线视频| 亚洲综合电影一区二区三区| 久久99青青| 国产精品毛片| 国产一区福利| 夜久久久久久| caoporn视频在线| 亚洲精品综合| 精品精品99| 激情久久久久久| 久久中文视频| 日韩电影免费网站| 亚洲一区二区三区无吗|