溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

JS前端的內(nèi)存處理的方法是什么

發(fā)布時(shí)間:2023-04-11 15:07:46 來(lái)源:億速云 閱讀:119 作者:iii 欄目:開(kāi)發(fā)技術(shù)

這篇文章主要介紹了JS前端的內(nèi)存處理的方法是什么的相關(guān)知識(shí),內(nèi)容詳細(xì)易懂,操作簡(jiǎn)單快捷,具有一定借鑒價(jià)值,相信大家閱讀完這篇JS前端的內(nèi)存處理的方法是什么文章都會(huì)有所收獲,下面我們一起來(lái)看看吧。

一、內(nèi)存的儲(chǔ)存和代碼執(zhí)行的場(chǎng)所關(guān)系

對(duì)于任何語(yǔ)言來(lái)說(shuō),內(nèi)存管理、垃圾回收等知識(shí)都是進(jìn)階路上繞不開(kāi)的坎。出來(lái)面試估計(jì)也沒(méi)少被問(wèn)到“前端的內(nèi)存處理你了解么? 你知道js中的垃圾處理機(jī)制嗎? 什么情況會(huì)導(dǎo)致內(nèi)存泄漏呢?

1. 儲(chǔ)存空間

兩種結(jié)構(gòu):

  • 棧空間:

  • 存儲(chǔ)原始類(lèi)型

  • 執(zhí)行上下文 (代碼空間:主要存儲(chǔ)可執(zhí)行代碼)

  • 堆空間:存儲(chǔ)引用類(lèi)型

為什么不都用棧存呢?

因?yàn)樾枰脳?lái)維護(hù)程序執(zhí)行期間上下文的狀態(tài),如果??臻g大了話,所有的數(shù)據(jù)都存放在??臻g里面,那么會(huì)影響到上下文切換的效率,進(jìn)而又影響到整個(gè)程序的執(zhí)行效率。

2. 內(nèi)存的生命周期

內(nèi)存分配:聲明變量、函數(shù)、對(duì)象的時(shí)候 內(nèi)存的使用:讀寫(xiě)內(nèi)存,使用變量 函數(shù)等 內(nèi)存回收:使用完畢,由垃圾回收機(jī)制自動(dòng)回收不再使用的內(nèi)存

3. js 中的內(nèi)存分配和使用

// 分配
const num = 123;  // 分配到棧
const str = 'sss';// 分配到棧
const obj = {};   // 分配到堆
// 使用
const a = 10;
console.log(a); // 使用

4. 調(diào)用棧下移ESP(記錄當(dāng)前執(zhí)行狀態(tài)的指針)

當(dāng)一個(gè)函數(shù)執(zhí)行結(jié)束之后,JavaScript 引擎會(huì)通過(guò)向下移動(dòng) ESP 來(lái)銷(xiāo)毀該函數(shù)保存在棧中的執(zhí)行上下文。 比方下面這個(gè)例子通過(guò)ESP 狀態(tài)來(lái)展示就如圖所示。

var foo1 = () => {
    console.log('foo1')
    foo2()
}
var foo2 = () => {
    console.log('foo2')
}
foo1()

JS前端的內(nèi)存處理的方法是什么

被ESP指針移開(kāi)后的函數(shù)作用域foo1 明顯屬于不在被引用,后續(xù)將會(huì)直接被GC回收

二、 js中的垃圾回收機(jī)制

有C語(yǔ)言經(jīng)驗(yàn)的開(kāi)發(fā)者,一定明白內(nèi)存聲明分配好之后,需要手動(dòng) free 的操作,這就是手動(dòng)回收。而 Js 本身是自動(dòng)回收機(jī)制,所以開(kāi)發(fā)者不需要過(guò)多關(guān)注內(nèi)存分配和釋放的問(wèn)題。這些工作都讓 V8 引擎中的垃圾回收器(GC)給承包了。

最早接觸 js 的時(shí)候,市面上對(duì)于 js 內(nèi)存管理、垃圾回收主要講的是下面兩種概念:

1. 引用計(jì)數(shù)法

引用計(jì)數(shù)法的算法主要依賴于引用的概念,這個(gè)回收機(jī)制最早是在 IE 在使用的。目前主流瀏覽器都使用標(biāo)記清除法了??匆粋€(gè)對(duì)象是否有指向他的引用,如果沒(méi)有其他對(duì)象指向他了,說(shuō)明當(dāng)前這個(gè)對(duì)象不再被需要了。

他的缺陷在于:循環(huán)引用

如果兩個(gè)對(duì)象相互引用,盡管他們已不再被使用,但是引用計(jì)數(shù)無(wú)法識(shí)別,導(dǎo)致內(nèi)存泄漏。

2. 標(biāo)記清除法(Mark-Sweep)

將“不再使用的的對(duì)象”定義為“無(wú)法到達(dá)的對(duì)象

從根部js的全局對(duì)象觸發(fā),定時(shí)遞歸掃描內(nèi)存中的對(duì)象,凡是無(wú)法從根部到達(dá)的對(duì)象,就會(huì)被標(biāo)記為不再使用,稍后進(jìn)行回收。

執(zhí)行過(guò)程如下:

  • GC在運(yùn)行的時(shí)候會(huì)給內(nèi)存中的所有變量都加上標(biāo)記

  • 將從根部觸發(fā)能夠觸及到的對(duì)象標(biāo)記清除

  • 剩下的還有標(biāo)記的變量被視為準(zhǔn)備刪除的變量

  • GC銷(xiāo)毀帶有標(biāo)記的值 回收內(nèi)存空間

三、代際假說(shuō)和分代收集

代際假說(shuō)(The Generational Hypothesis)是現(xiàn)代瀏覽器垃圾回收策略的基礎(chǔ)。整個(gè)模型可以看看下圖

JS前端的內(nèi)存處理的方法是什么

新生代(副垃圾回收器)

存放的是生存時(shí)間短、占用空間較小的的對(duì)象,通過(guò) Scavenge 算法,是把新生代空間對(duì)半劃分為兩個(gè)區(qū)域,一半是對(duì)象區(qū)域,一半是空閑區(qū)域。新的對(duì)象都要放到對(duì)象區(qū),當(dāng)快滿的時(shí)候,將還存活的對(duì)象復(fù)制到空閑區(qū)后進(jìn)行角色互換。并且執(zhí)行對(duì)象晉升策略,對(duì)象區(qū)域和空閑區(qū)域翻轉(zhuǎn)兩次還存在的對(duì)象,升級(jí)到老生代。這個(gè)復(fù)制翻轉(zhuǎn)的過(guò)程也避免內(nèi)存碎片的產(chǎn)生。

老生代(主垃圾回收器)

存放的生存時(shí)間久的對(duì)象或者大的對(duì)象,使用標(biāo)記清除的算法進(jìn)行垃圾回收。一旦執(zhí)行垃圾回收算法,都需要將正在執(zhí)行的 JavaScript 腳本暫停下來(lái),待垃圾回收完畢后再恢復(fù)腳本執(zhí)行。這種行為稱(chēng)為全停頓(Stop-The-World)。實(shí)際上瀏覽器為了避免垃圾回收卡頓通過(guò)增量標(biāo)記方式將回收任務(wù)拆解成多個(gè)小任務(wù)穿插在js主線程中執(zhí)行。

四、常見(jiàn)的內(nèi)存泄漏

1. 全局變量

function foo() {
    bar1 = 'aaa'; // 相當(dāng)于聲明在window.bar1
    this.bar2 = 'aaaa'
}
foo(); // 執(zhí)行函數(shù)事this指向window ,相當(dāng)于一個(gè)函數(shù)給全局變量增加了兩個(gè)變量

2. 未被清理的定時(shí)器和回調(diào)函數(shù)

//setInterval
//setTimeout
setInterval(() => {
    console.log('test')
}, 500)
//沒(méi)用用 clearInterval clearTimeout 做清除

3. 閉包

個(gè)人最喜歡《你不知道的js》里對(duì)閉包的描述

一個(gè)內(nèi)部函數(shù),有權(quán)訪問(wèn)包含其的外部函數(shù)的變量 —— 《你不知道的js》 或者也可以用“內(nèi)存逃逸”這種高逼格的屬于形容。

// 閉包 gc 案例
var one = null;
var replace = function() {
    var originalOne = one;
    var unused =function() {
        if(originalOne) {
            console.log(111);
        }
    }
    one = {
        longString: '111',
        method: function() {
            console.log()
        }
    }    
}
setInterval(replace, 500)

每次調(diào)用 replace, one 得到一個(gè)包含字符串和一個(gè)對(duì)于新閉包 method 的對(duì)象 unused 引用了 originOne

5. DOM 引用

var elements= {
    image: document.getElementById('111');
}
elements.image = null;

6. 怎么避免呢?

  • 盡量減少全局變量

//盡可能少寫(xiě)
window.object = {} // 這類(lèi)代碼
  • 使用完引用數(shù)據(jù)后,及時(shí)解除引用.null

let obj = {}
...
obj = null;
  • 避免死循環(huán)等持續(xù)執(zhí)行的操作(例如 邊界判斷不清晰的 for 或 while 循環(huán))

  • 多使用 WeakSet / WeakMap 特性

// Vue3 中就大量用了WeakMap 優(yōu)化實(shí)例引用
const bucket = new WeakMap();
...
const obj = new Proxy(data, {
  get(target, key) {
    track(target, key);
    return target[key];
  },
  set(target, key, newVal) {
    target[key] = newVal;
    trigger(target, key);
  },
});
function track(target, key) {
  if (!activeEffect) return;
  let depsMap = bucket.get(target);
  if (!depsMap) {
    bucket.set(target, (depsMap = new Map()));
  }
  // 再根據(jù)key 從 depsMap 中取得 deps, 它是一個(gè) Set 類(lèi)型,里面存儲(chǔ)著左右與當(dāng)前 key 相關(guān)聯(lián)的副作用函數(shù): effects
  let deps = depsMap.get(key);
  // 如果 deps 不存在,同樣新建一個(gè) Set 并與 key 關(guān)聯(lián)
  if (!deps) {
    depsMap.set(key, (deps = new Set()));
  }
  deps.add(activeEffect);
  activeEffect.deps.push(deps);
}
...

關(guān)于“JS前端的內(nèi)存處理的方法是什么”這篇文章的內(nèi)容就介紹到這里,感謝各位的閱讀!相信大家對(duì)“JS前端的內(nèi)存處理的方法是什么”知識(shí)都有一定的了解,大家如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

js
AI