前端常見內(nèi)存泄漏

內(nèi)存泄漏

系統(tǒng)進(jìn)程不再用到的內(nèi)存,沒有及時(shí)釋放,就叫做內(nèi)存泄漏(memory leak)。當(dāng)內(nèi)存占用越來越高,輕則影響系統(tǒng)性能,重則導(dǎo)致進(jìn)程崩潰。Chrome 限制了瀏覽器所能使用的內(nèi)存極限(64 位為 1.4GB,32 位為 1.0GB)

一、引起內(nèi)存泄漏的原因

1. 意外的全局變量

由于 js 對(duì)未聲明變量的處理方式是在全局對(duì)象上創(chuàng)建該變量的引用。如果在瀏覽器中,全局對(duì)象就是 window 對(duì)象。變量在窗口關(guān)閉或重新刷新頁面之前都不會(huì)被釋放,如果未聲明的變量緩存大量的數(shù)據(jù),就會(huì)導(dǎo)致內(nèi)存泄露。

1.1 未聲明變量

function fn() {
  a = 'hello'
}
fn()

1.2 使用 this 創(chuàng)建的變量(this 的指向是 window)。

function fn() {
  this.a = 'hello'
}
fn()

解決方法:

  • 避免創(chuàng)建全局變量
  • 使用嚴(yán)格模式,在 JavaScript 文件頭部或者函數(shù)的頂部加上 use strict

2. 閉包引起的內(nèi)存泄漏

由于閉包可以讀取函數(shù)內(nèi)部的變量,然后讓這些變量始終保存在內(nèi)存中。如果在使用結(jié)束后沒有將局部變量清除,就可能導(dǎo)致內(nèi)存泄露。

function fn () {
  var a = "hello"
  return function () {
    console.log(a)
  }
}

解決方法:將事件處理函數(shù)定義在外部,解除閉包,或者在定義事件處理函數(shù)的外部函數(shù)中。
比如:在循環(huán)中的函數(shù)表達(dá)式,能復(fù)用最好放到循環(huán)外面。

// bad
for (var k = 0; k < 5; k++) {
  var t = function (a) {
    // 創(chuàng)建了5次函數(shù)對(duì)象。
    console.log(a)
  }
  t(k)
}

// good
function t(a) {
  console.log(a)
}
for (var k = 0; k < 10; k++) {
  t(k)
}
t = null

3. 沒有清理的DOM元素引用

雖然在某個(gè)地方刪除了元素,但是對(duì)象中還存在對(duì)dom的引用。

// 在對(duì)象中引用DOM
var elements = {
  btn: document.getElementById('btn'),
}
function doSomeThing() {
  elements.btn.click()
}

function removeBtn() {
  // 將body中的btn移除, 也就是移除 DOM樹中的btn
  document.body.removeChild(document.getElementById('btn'))
  // 但是此時(shí)全局變量elements還是保留了對(duì)btn的引用, btn還是存在于內(nèi)存中,不能被回收
}

解決方法:手動(dòng)刪除,elements.btn = null

4. 被遺忘的定時(shí)器或者回調(diào)

定時(shí)器中有 dom 的引用,即使 dom 刪除了,但是定時(shí)器還在,所以內(nèi)存中還是有這個(gè) dom。

// 定時(shí)器 loadData例為請(qǐng)求數(shù)據(jù)函數(shù)
var serverData = loadData()
setInterval(function () {
  var renderer = document.getElementById('renderer')
  if (renderer) {
    renderer.innerHTML = JSON.stringify(serverData)
  }
}, 5000)

// 觀察者模式
var btn = document.getElementById('btn')
function onClick(element) {
  element.innerHTMl = "innerHTML"
}
btn.addEventListener('click', onClick)

解決方法:

  • 手動(dòng)刪除定時(shí)器和 dom
  • removeEventListener 移除事件監(jiān)聽

二、vue中容易出現(xiàn)內(nèi)存泄漏的幾種情況

在 Vue單頁面開發(fā)應(yīng)用,那么就更要當(dāng)心內(nèi)存泄漏的問題。因?yàn)樵?SPA 的設(shè)計(jì)中,用戶使用它是不需要刷新瀏覽器的,所以JavaScript應(yīng)用需要自行清理組件來確保垃圾回收以預(yù)期的方式生效。因此開發(fā)過程中,需要時(shí)刻警惕內(nèi)存泄漏的問題。

1.全局變量造成的內(nèi)存泄露

聲明的全局變量在切換頁面的時(shí)候沒有清空

<template>
  <div id="home">這里是首頁</div>
</template>
<script>
  export default {
    mounted() {
      window.test = {
        // 此處在全局window對(duì)象中引用了本頁面的dom對(duì)象
        name: 'home',
        node: document.getElementById('home'),
      }
    },
  }
</script>

解決方法:在頁面卸載的時(shí)候順便處理掉該引用

destroyed () {
  window.test = null // 頁面卸載的時(shí)候解除引用
 }

2. 監(jiān)聽在 window/body 等事件沒有解綁

特別注意 window.addEventListener 之類的時(shí)間監(jiān)聽

<template>
  <div id="home">這里是首頁</div>
</template>

<script>
export default {
  mounted () {
    window.addEventListener('resize', this.func) // window對(duì)象引用了home頁面的方法
  }
}
</script>

解決方法:在頁面銷毀的時(shí)候,順便解除引用,釋放內(nèi)存

beforeDestroy () {
  window.removeEventListener('resize', this.func)
}

3. 綁在 EventBus 的事件沒有解綁

<template>
  <div id="home">這里是首頁</div>
</template>

<script>
export default {
  mounted () {
   this.$EventBus.$on('homeTask', res => this.func(res))
  }
}
</script>

解決方法:在頁面卸載的時(shí)候也可以考慮解除引用

mounted () {
 this.$EventBus.$on('homeTask', res => this.func(res))
},
destroyed () {
 this.$EventBus.$off()
}

4.Echarts

每一個(gè)圖例在沒有數(shù)據(jù)的時(shí)候它會(huì)創(chuàng)建一個(gè)定時(shí)器去渲染氣泡,頁面切換后,echarts 圖例是銷毀了,但是這個(gè) echarts 的實(shí)例還在內(nèi)存當(dāng)中,同時(shí)它的氣泡渲染定時(shí)器還在運(yùn)行。這就導(dǎo)致 Echarts 占用 CPU 高,導(dǎo)致瀏覽器卡頓,當(dāng)數(shù)據(jù)量比較大時(shí)甚至瀏覽器崩潰。
解決方法:加一個(gè) beforeDestroy()方法釋放該頁面的 chart 資源。
clear():清空繪畫內(nèi)容,清空后實(shí)例可用,因?yàn)椴⒎轻尫攀纠馁Y源,釋放資源我們需要dispose()。
dispose():釋放圖表實(shí)例,釋放后實(shí)例不再可用。

beforeDestroy () {
  this.chart.clear()
  this.chart.dispose()
}

5. v-if 指令產(chǎn)生的內(nèi)存泄露

v-if 綁定到 false 的值,但是實(shí)際上 dom 元素在隱藏的時(shí)候沒有被真實(shí)的釋放掉。
比如下面的示例中,我們加載了一個(gè)帶有非常多選項(xiàng)的選擇框,然后我們用到了一個(gè)顯示/隱藏按鈕,通過一個(gè) v-if 指令從虛擬 DOM 中添加或移除它。這個(gè)示例的問題在于這個(gè) v-if 指令會(huì)從 DOM 中移除父級(jí)元素,但是我們并沒有清除由 Choices.js 新添加的 DOM 片段,從而導(dǎo)致了內(nèi)存泄漏。

<template>
<div id="app">
  <button v-if="showChoices" @click="hide">Hide</button>
  <button v-if="!showChoices" @click="show">Show</button>
  <div v-if="showChoices">
    <select id="choices-single-default"></select>
  </div>
</div>
</template>

<script>
  export default {
    data() {
      return {
        showChoices: true,
      }
    },
    mounted: function () {
      this.initializeChoices()
    },
    methods: {
      initializeChoices: function () {
        let list = []
        // 我們來為選擇框載入很多選項(xiàng),這樣的話它會(huì)占用大量的內(nèi)存
        for (let i = 0; i < 1000; i++) {
          list.push({
            label: 'Item ' + i,
            value: i,
          })
        }
        new Choices('#choices-single-default', {
          searchEnabled: true,
          removeItemButton: true,
          choices: list,
        })
      },
      show: function () {
        this.showChoices = true
        this.$nextTick(() => {
          this.initializeChoices()
        })
      },
      hide: function () {
        this.showChoices = false
      },
    },
  }
</script>

在上述的示例中,我們可以用 hide() 方法在將選擇框從 DOM 中移除之前做一些清理工作,來解決內(nèi)存泄露問題。為了做到這一點(diǎn),我們會(huì)在 Vue 實(shí)例的數(shù)據(jù)對(duì)象中保留一個(gè)屬性,并會(huì)使用 Choices API 中的 destroy() 方法將其清除。

<div id="app">
  <button v-if="showChoices" @click="hide">Hide</button>
  <button v-if="!showChoices" @click="show">Show</button>
  <div v-if="showChoices">
    <select id="choices-single-default"></select>
  </div>
</div>

<script>
  export default {
    data() {
      return {
        showChoices: true,
        choicesSelect: null
      }
    },
    mounted: function () {
      this.initializeChoices()
    },
    methods: {
      initializeChoices: function () {
        let list = []
        for (let i = 0; i < 1000; i++) {
          list.push({
            label: 'Item ' + i,
            value: i,
          })
        }
         // 在我們的 Vue 實(shí)例的數(shù)據(jù)對(duì)象中設(shè)置一個(gè) `choicesSelect` 的引用
        this.choicesSelect = new Choices("#choices-single-default", {
          searchEnabled: true,
          removeItemButton: true,
          choices: list,
        })
      },
      show: function () {
        this.showChoices = true
        this.$nextTick(() => {
          this.initializeChoices()
        })
      },
      hide: function () {
        // 現(xiàn)在我們可以讓 Choices 使用這個(gè)引用,從 DOM 中移除這些元素之前進(jìn)行清理工作
        this.choicesSelect.destroy()
        this.showChoices = false
      },
    },
  }
</script>

三、ES6 防止內(nèi)存泄漏

前面說過,及時(shí)清除引用非常重要。但是,有時(shí)候可能一疏忽就忘了,所以才有那么多內(nèi)存泄漏。
ES6考慮到這點(diǎn),推出了兩種新的數(shù)據(jù)結(jié)構(gòu):weakset 和 weakmap 。他們對(duì)值的引用都是不計(jì)入垃圾回收機(jī)制的,也就是說,如果其他對(duì)象都不再引用該對(duì)象,那么垃圾回收機(jī)制會(huì)自動(dòng)回收該對(duì)象所占用的內(nèi)存。

const wm = new WeakMap()
const element = document.getElementById('example')
vm.set(element, 'something')
vm.get(element)

上面代碼中,先新建一個(gè) Weakmap 實(shí)例。然后,將一個(gè)DOM節(jié)點(diǎn)作為鍵名存入該實(shí)例,并將一些附加信息作為鍵值,一起存放在WeakMap里面。這時(shí),WeakMap里面對(duì)element的引用就是弱引用,不會(huì)被計(jì)入垃圾回收機(jī)制。
注冊(cè)監(jiān)聽事件的 listener 對(duì)象很適合用 WeakMap來實(shí)現(xiàn)。

// 代碼1
ele.addEventListener('click', handler, false)

// 代碼2
const listener = new WeakMap()
listener.set(ele, handler)
ele.addEventListener('click', listener.get(ele), false)

代碼2比起代碼1的好處是:由于監(jiān)聽函數(shù)是放在WeakMap里面,一旦 dom對(duì)象ele消失,與它綁定的監(jiān)聽函數(shù)handler也會(huì)自動(dòng)消失。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容