Vue實(shí)現(xiàn)定位并解決內(nèi)存泄漏
Vue.js 是一個(gè)流行且強(qiáng)大的 JavaScript 框架,它允許我們構(gòu)建動(dòng)態(tài)和交互式 Web 應(yīng)用程序。
然而,與任何軟件一樣,Vue.js 應(yīng)用程序有時(shí)會(huì)遇到內(nèi)存泄漏,從而導(dǎo)致性能下降和意外行為。
今天,我們將深入探討 Vue.js 應(yīng)用程序中內(nèi)存泄漏的原因,并探索如何定位和修復(fù)這些問題的有效策略。
什么是內(nèi)存泄漏
當(dāng)程序執(zhí)行過程中保留不再需要的內(nèi)存時(shí)(主要是一些 變量、 方法等),會(huì)阻止內(nèi)存被釋放并導(dǎo)致程序的內(nèi)存使用量隨著時(shí)間的推移而增長,稱為內(nèi)存泄漏。
在 Vue.js 應(yīng)用程序中,內(nèi)存泄漏通常是由于組件、全局 EventBus、事件定時(shí)器 和 變量,函數(shù)引用的管理不當(dāng)而引起的。
1. EventBus 引起的內(nèi)存泄露
一個(gè)老生常談的話題, Vue.js 中跨組件通信,要么是EventBus, 要么是Vuex 或者Pinia 這種數(shù)據(jù)流工具。
當(dāng)我們對(duì) EventBus 使用不當(dāng)時(shí),它們可能導(dǎo)致內(nèi)存泄漏。
當(dāng)組件被銷毀時(shí),應(yīng)將它們從事件總線中刪除,以防止延遲引用。
舉個(gè)例子:
// 組件A.vue
<template>
<div>
<button @click="sendMessage">廣播消息</button>
</div>
</template>
<script>
import { EventBus } from "./EventBus.js";
export default {
methods: {
sendMessage() {
EventBus.$emit("message", "Hello world from A!");
}
}
};
</script>
// 組件B.vue
<template>
<div>
<p>{{ receivedMessage }}</p>
</div>
</template>
<script>
import { EventBus } from "./EventBus.js";
export default {
data() {
return {
receivedMessage: ""
};
},
created() {
EventBus.$on("message", message => {
this.receivedMessage = message;
});
}
};
</script>在此示例中,發(fā)生內(nèi)存泄漏是因?yàn)?ComponentB 從EventBus訂閱了一個(gè)事件,但在該組件被銷毀時(shí)并未取消訂閱。
為了解決這個(gè)問題,我們需要在 組件B 的 beforeDestroy 鉤子中使用 EventBus.$off 來刪除事件監(jiān)聽器。
所以 需要對(duì)組件B做如下修改:
// ComponentB.vue
<template>
<div>
<p>{{ receivedMessage }}</p>
</div>
</template>
<script>
import { EventBus } from "./EventBus.js";
export default {
data() {
return {
receivedMessage: ""
};
},
created() {
EventBus.$on("message", message => {
this.receivedMessage = message;
});
},
beforeDestroy() {
EventBus.$off("message"); //this line was missing previously
}
};
</script>2. 未被清理的定時(shí)器
Vue.js 應(yīng)用程序中內(nèi)存泄漏的最常見原因之一是未能正確刪除定時(shí)器。當(dāng)組件在其生命周期中使用了定時(shí)器,但并未合理的進(jìn)行清除。
一旦組件被銷毀, 定時(shí)器會(huì)繼續(xù)引用該組件,從而防止其被垃圾收集。
舉個(gè)例子:
<template>
<div>
<button @click="startShow">開始演示</button>
<button @click="stopShow">停止演示</button>
</div>
</template>
<script>
export default {
data() {
return {
intervalId: null
};
},
methods: {
startLeak() {
this.intervalId = setInterval(() => {
// Simulate some activity
console.log("Interval running...");
}, 1000);
},
stopLeak() {
clearInterval(this.intervalId);
this.intervalId = null;
}
}
};
</script>在這個(gè)例子中,發(fā)生內(nèi)存泄漏是因?yàn)閱螕?“開始演示” 按鈕時(shí)創(chuàng)建了interval 定時(shí),但在組件被銷毀時(shí)沒有正確清理它。
為了解決這個(gè)問題,我們需要在 beforeDestroy 生命周期中,對(duì)定時(shí)器進(jìn)行清理。
所以最終的代碼將如下所示:
<template>
<div>
<button @click="startShow">開始演示</button>
<button @click="stopShow">停止演示</button>
</div>
</template>
<script>
export default {
data() {
return {
intervalId: null
};
},
methods: {
startLeak() {
this.intervalId = setInterval(() => {
// Simulate some activity
console.log("Interval running...");
}, 1000);
},
stopLeak() {
clearInterval(this.intervalId);
this.intervalId = null;
}
},
@diff new
beforeDestroy() {
clearInterval(this.intervalId); // This line is missing above
}
};
</script>3. 第三方庫使用不當(dāng)
第三方類庫使用不當(dāng),是內(nèi)存泄漏的最常見原因。
這是由于組件清理不當(dāng)造成的。這里我使用 Choices.js 庫進(jìn)行演示。
// cdn Choice Library
<link rel='stylesheet prefetch' >
<script src='https://joshuajohnson.co.uk/Choices/assets/scripts/dist/choices.min.js?version=3.0.3'></script>
// our component
<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
new Vue({
el: "#app",
data: function () {
return {
showChoices: true
}
},
mounted: function () {
this.initializeChoices()
},
methods: {
initializeChoices: function () {
let list = []
// 創(chuàng)造更多的選項(xiàng),方便直觀的觀察到內(nèi)存泄漏
for (let i = 0; i < 1000; i++) {
list.push({
label: "選項(xiàng)" + 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
}
}
})在上面的例子中,
我們加載了一個(gè)包含許多選項(xiàng)的下拉列表,然后使用帶有 v-if 指令的顯示/隱藏按鈕來添加它并將其從虛擬 DOM 中刪除。
此示例的問題在于 v-if 指令從 DOM 中刪除了父元素,但我們沒有清理 Choices.js 創(chuàng)建的額外 DOM 片段,從而導(dǎo)致內(nèi)存泄漏。
我為大家做了一個(gè)在線的示例工程,便于大家去觀察內(nèi)存占用情況。
http://demolab.seanz.net/m-leak/index.html
當(dāng)多點(diǎn)幾次 之后,會(huì)有明確的內(nèi)存增加。 附上一張 差異圖。
初始化狀態(tài):

多次點(diǎn)擊狀態(tài):

識(shí)別內(nèi)存泄漏
識(shí)別 Vue.js 應(yīng)用程序中的內(nèi)存泄漏可能具有挑戰(zhàn)性,因?yàn)樗鼈兺ǔ1憩F(xiàn)為性能緩慢或隨著時(shí)間的推移內(nèi)存消耗增加。沒有神奇的工具可以識(shí)別代碼的問題所在。
但是,大多數(shù)現(xiàn)代瀏覽器都提供內(nèi)存分析工具,允許您拍攝應(yīng)用程序隨時(shí)間的內(nèi)存使用情況的快照。這些工具可以幫助您識(shí)別哪些對(duì)象消耗了過多的內(nèi)存以及哪些組件沒有得到正確的垃圾收集。
Chrome 的 "Heap Snapshot" 等工具可以通過可視化對(duì)象引用及其內(nèi)存消耗來提供對(duì)內(nèi)存使用情況的詳細(xì)了解。
這可以幫助您更準(zhǔn)確地查明內(nèi)存泄漏的根源。
如何盡可能避免內(nèi)存泄漏
- 正確的使用計(jì)時(shí)器:確保在 Mounted 生命周期掛鉤期間添加 計(jì)時(shí)器,在組件的 beforeDestroy 鉤子中,進(jìn)行清除。
- EventBus :當(dāng)你在組件中使用EventBus時(shí),一定要確保,在組件銷毀鉤子方法中 將其 從 EventBus 中刪除。
- 響應(yīng)式數(shù)據(jù)清理:在 beforeDestroy 生命周期鉤子中,清理響應(yīng)式數(shù)據(jù)屬性,以防止它們保留對(duì)已銷毀組件的引用。
- 第三方類庫:當(dāng)使用在 Vue 之外操作 DOM 的其他 3rd 方庫時(shí),通常會(huì)發(fā)生這些泄漏。要修復(fù)此類泄漏,請(qǐng)正確遵循庫文檔并采取適當(dāng)?shù)拇胧?/li>
總結(jié)
Vue.js 應(yīng)用程序中的內(nèi)存泄漏和性能測試可能很難識(shí)別和解決,而且在快速交付的興奮中也很容易被忽視。
但是,保持較小的內(nèi)存占用對(duì)于整體用戶體驗(yàn)仍然很重要。
借助正確的工具、技術(shù)和實(shí)踐,您可以顯著減少遇到它們的機(jī)會(huì)。
通過正確管理一些明顯引起內(nèi)存泄漏的方法,您可以確保 Vue.js 應(yīng)用程序以最佳性能運(yùn)行并保持健康的內(nèi)存占用。
以上就是Vue實(shí)現(xiàn)定位并解決內(nèi)存泄漏的詳細(xì)內(nèi)容,更多關(guān)于Vue內(nèi)存泄漏的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
vue實(shí)現(xiàn)input輸入模糊查詢的三種方式
本文主要介紹了vue實(shí)現(xiàn)input輸入模糊查詢的三種方式嗎,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
詳解Webstorm 新建.vue文件支持高亮vue語法和es6語法
這篇文章主要介紹了Webstorm 添加新建.vue文件功能并支持高亮vue語法和es6語法,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-10-10
淺談vue實(shí)現(xiàn)數(shù)據(jù)監(jiān)聽的函數(shù) Object.defineProperty
本篇文章主要介紹了淺談vue實(shí)現(xiàn)數(shù)據(jù)監(jiān)聽的函數(shù) Object.defineProperty,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
解決vite+vue3項(xiàng)目打包后圖片不顯示或者請(qǐng)求路徑多了一個(gè)undefined問題
這篇文章主要介紹了解決vite+vue3項(xiàng)目打包后圖片不顯示或者請(qǐng)求路徑多了一個(gè)undefined問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2025-05-05
vue實(shí)現(xiàn)移動(dòng)端拖動(dòng)排序
這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)移動(dòng)端拖動(dòng)排序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-08-08
Nuxt pages下不同的頁面對(duì)應(yīng)layout下的頁面布局操作
這篇文章主要介紹了Nuxt pages下不同的頁面對(duì)應(yīng)layout下的頁面布局操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-11-11
vue 全選與反選的實(shí)現(xiàn)方法(無Bug 新手看過來)
下面小編就為大家分享一篇vue 全選與反選的實(shí)現(xiàn)方法(無Bug 新手看過來),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-02-02
安裝vue3開發(fā)者工具但控制臺(tái)沒有顯示出vue選項(xiàng)的解決
這篇文章主要介紹了安裝vue3開發(fā)者工具但控制臺(tái)沒有顯示出vue選項(xiàng)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10
VuePress整合Toast消息提示的實(shí)現(xiàn)步驟
本文主要介紹了VuePress整合Toast消息提示的實(shí)現(xiàn)步驟,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2025-06-06

