真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

JavaScript閉包用多會造成內(nèi)存泄露嗎

今天小編給大家分享一下JavaScript閉包用多會造成內(nèi)存泄露嗎的相關(guān)知識點,內(nèi)容詳細(xì),邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

目前成都創(chuàng)新互聯(lián)公司已為千余家的企業(yè)提供了網(wǎng)站建設(shè)、域名、虛擬空間、成都網(wǎng)站托管、企業(yè)網(wǎng)站設(shè)計、耿馬網(wǎng)站維護(hù)等服務(wù),公司將堅持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。

場景思考

以下案例: 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)存大?。?/p>

JavaScript閉包用多會造成內(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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

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

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

JavaScript閉包用多會造成內(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)存應(yīng)該都被釋放掉了啊?

我們繼續(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)存回收機(jī)制

閉包簡介

閉包:一個函數(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,導(dǎo)致變量 num 滯留在內(nèi)存中

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

JavaScript閉包用多會造成內(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)存里表示如下

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

再定義一個函數(shù)

let a = '小馬哥'function fn() {  let num = 1}

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

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

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

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

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

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

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

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

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

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

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

JavaScript閉包用多會造成內(nèi)存泄露嗎

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

JavaScript閉包用多會造成內(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);
 };
};

以上就是“JavaScript閉包用多會造成內(nèi)存泄露嗎”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學(xué)習(xí)更多的知識,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


新聞名稱:JavaScript閉包用多會造成內(nèi)存泄露嗎
轉(zhuǎn)載來于:http://weahome.cn/article/ggcpph.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部