前端常见内存泄漏及解决方案总结

小鱼儿 2022-10-25 11:25 384阅读 0赞

##引起内存泄漏的原因
意外的全局变量
由于 js 对未声明变量的处理方式是在全局对象上创建该变量的引用。如果在浏览器中,全局对象就是 window 对象。变量在窗口关闭或重新刷新页面之前都不会被释放,如果未声明的变量缓存大量的数据,就会导致内存泄露。

未声明变量

  1. function fn() {
  2. a = 'global variable'
  3. }
  4. fn()

使用 this 创建的变量(this 的指向是 window)。

  1. function fn() {
  2. this.a = 'global variable'
  3. }
  4. fn()

解决方法:
避免创建全局变量
使用严格模式,在 JavaScript 文件头部或者函数的顶部加上 use strict。
闭包引起的内存泄漏
原因:闭包可以读取函数内部的变量,然后让这些变量始终保存在内存中。如果在使用结束后没有将局部变量清除,就可能导致内存泄露。

  1. function fn () {
  2. var a = "I'm a";
  3. return function () {
  4. console.log(a);
  5. };
  6. }

解决:将事件处理函数定义在外部,解除闭包,或者在定义事件处理函数的外部函数中。
比如:在循环中的函数表达式,能复用最好放到循环外面。

  1. // bad
  2. for (var k = 0; k < 10; k++) {
  3. var t = function (a) {
  4. // 创建了10次 函数对象。
  5. console.log(a)
  6. }
  7. t(k)
  8. }
  9. // good
  10. function t(a) {
  11. console.log(a)
  12. }
  13. for (var k = 0; k < 10; k++) {
  14. t(k)
  15. }
  16. t = null

没有清理的 DOM 元素引用
原因:虽然别的地方删除了,但是对象中还存在对 dom 的引用。

  1. // 在对象中引用DOM
  2. var elements = {
  3. btn: document.getElementById('btn'),
  4. }
  5. function doSomeThing() {
  6. elements.btn.click()
  7. }
  8. function removeBtn() {
  9. // 将body中的btn移除, 也就是移除 DOM树中的btn
  10. document.body.removeChild(document.getElementById('button'))
  11. // 但是此时全局变量elements还是保留了对btn的引用, btn还是存在于内存中,不能被GC回收
  12. }

解决方法:手动删除,elements.btn = null。
被遗忘的定时器或者回调
定时器中有 dom 的引用,即使 dom 删除了,但是定时器还在,所以内存中还是有这个 dom。

  1. var serverData = loadData()
  2. setInterval(function () {
  3. var renderer = document.getElementById('renderer')
  4. if (renderer) {
  5. renderer.innerHTML = JSON.stringify(serverData)
  6. }
  7. }, 5000)
  8. // 观察者模式
  9. var btn = document.getElementById('btn')
  10. function onClick(element) {
  11. element.innerHTMl = "I'm innerHTML"
  12. }
  13. btn.addEventListener('click', onClick)

解决方法:
手动删除定时器和 dom。
removeEventListener 移除事件监听
vue 中容易出现内存泄露的几种情况
在 Vue SPA 开发应用,那么就更要当心内存泄漏的问题。因为在 SPA 的设计中,用户使用它时是不需要刷新浏览器的,所以 JavaScript 应用需要自行清理组件来确保垃圾回收以预期的方式生效。因此开发过程中,你需要时刻警惕内存泄漏的问题
全局变量造成的内存泄露
声明的全局变量在切换页面的时候没有清空

  1. <template>
  2. <div id="home">这里是首页</div>
  3. </template>
  4. <script>
  5. export default {
  6. mounted() {
  7. window.test = {
  8. // 此处在全局window对象中引用了本页面的dom对象
  9. name: 'home',
  10. node: document.getElementById('home'),
  11. }
  12. },
  13. }
  14. </script>

解决方案:在页面卸载的时候顺便处理掉该引用。

  1. destroyed () {
  2. window.test = null // 页面卸载的时候解除引用
  3. }

监听在 window/body 等事件没有解绑
特别注意 window.addEventListener 之类的时间监听

  1. <template>
  2. <div id="home">这里是首页</div>
  3. </template>
  4. <script>
  5. export default {
  6. mounted () {
  7. window.addEventListener('resize', this.func) // window对象引用了home页面的方法
  8. }
  9. }
  10. </script>

解决方法:在页面销毁的时候,顺便解除引用,释放内存

  1. mounted () {
  2. window.addEventListener('resize', this.func)
  3. },
  4. beforeDestroy () {
  5. window.removeEventListener('resize', this.func)
  6. }

绑在 EventBus 的事件没有解绑

  1. <template>
  2. <div id="home">这里是首页</div>
  3. </template>
  4. <script>
  5. export default {
  6. mounted () {
  7. this.$EventBus.$on('homeTask', res => this.func(res))
  8. }
  9. }
  10. </script>

解决方法:在页面卸载的时候也可以考虑解除引用

  1. mounted () {
  2. this.$EventBus.$on('homeTask', res => this.func(res))
  3. },
  4. destroyed () {
  5. this.$EventBus.$off()
  6. }

Echarts
每一个图例在没有数据的时候它会创建一个定时器去渲染气泡,页面切换后,echarts 图例是销毁了,但是这个 echarts 的实例还在内存当中,同时它的气泡渲染定时器还在运行。这就导致 Echarts 占用 CPU 高,导致浏览器卡顿,当数据量比较大时甚至浏览器崩溃。
解决方法:加一个 beforeDestroy()方法释放该页面的 chart 资源,我也试过使用 dispose()方法,但是 dispose 销毁这个图例,图例是不存在了,但图例的 resize()方法会启动,则会报没有 resize 这个方法,而 clear()方法则是清空图例数据,不影响图例的 resize,而且能够释放内存,切换的时候就很顺畅了。

  1. beforeDestroy () {
  2. this.chart.clear()
  3. }

v-if 指令产生的内存泄露
v-if 绑定到 false 的值,但是实际上 dom 元素在隐藏的时候没有被真实的释放掉。
比如下面的示例中,我们加载了一个带有非常多选项的选择框,然后我们用到了一个显示/隐藏按钮,通过一个 v-if 指令从虚拟 DOM 中添加或移除它。这个示例的问题在于这个 v-if 指令会从 DOM 中移除父级元素,但是我们并没有清除由 Choices.js 新添加的 DOM 片段,从而导致了内存泄漏。

  1. <div id="app">
  2. <button v-if="showChoices" @click="hide">Hide</button>
  3. <button v-if="!showChoices" @click="show">Show</button>
  4. <div v-if="showChoices">
  5. <select id="choices-single-default"></select>
  6. </div>
  7. </div>
  8. <script>
  9. export default {
  10. data() {
  11. return {
  12. showChoices: true,
  13. }
  14. },
  15. mounted: function () {
  16. this.initializeChoices()
  17. },
  18. methods: {
  19. initializeChoices: function () {
  20. let list = []
  21. // 我们来为选择框载入很多选项,这样的话它会占用大量的内存
  22. for (let i = 0; i < 1000; i++) {
  23. list.push({
  24. label: 'Item ' + i,
  25. value: i,
  26. })
  27. }
  28. new Choices('#choices-single-default', {
  29. searchEnabled: true,
  30. removeItemButton: true,
  31. choices: list,
  32. })
  33. },
  34. show: function () {
  35. this.showChoices = true
  36. this.$nextTick(() => {
  37. this.initializeChoices()
  38. })
  39. },
  40. hide: function () {
  41. this.showChoices = false
  42. },
  43. },
  44. }
  45. </script>

在上述的示例中,我们可以用 hide() 方法在将选择框从 DOM 中移除之前做一些清理工作,来解决内存泄露问题。为了做到这一点,我们会在 Vue 实例的数据对象中保留一个属性,并会使用 Choices API 中的 destroy() 方法将其清除。

  1. <div id="app">
  2. <button v-if="showChoices" @click="hide">Hide</button>
  3. <button v-if="!showChoices" @click="show">Show</button>
  4. <div v-if="showChoices">
  5. <select id="choices-single-default"></select>
  6. </div>
  7. </div>
  8. <script>
  9. export default {
  10. data() {
  11. return {
  12. showChoices: true,
  13. choicesSelect: null
  14. }
  15. },
  16. mounted: function () {
  17. this.initializeChoices()
  18. },
  19. methods: {
  20. initializeChoices: function () {
  21. let list = []
  22. for (let i = 0; i < 1000; i++) {
  23. list.push({
  24. label: 'Item ' + i,
  25. value: i,
  26. })
  27. }
  28. // 在我们的 Vue 实例的数据对象中设置一个 `choicesSelect` 的引用
  29. this.choicesSelect = new Choices("#choices-single-default", {
  30. searchEnabled: true,
  31. removeItemButton: true,
  32. choices: list,
  33. })
  34. },
  35. show: function () {
  36. this.showChoices = true
  37. this.$nextTick(() => {
  38. this.initializeChoices()
  39. })
  40. },
  41. hide: function () {
  42. // 现在我们可以让 Choices 使用这个引用,从 DOM 中移除这些元素之前进行清理工作
  43. this.choicesSelect.destroy()
  44. this.showChoices = false
  45. },
  46. },
  47. }
  48. </script>

ES6 防止内存泄漏
及时清除引用非常重要。但是不可能记得那么多,有时候一疏忽就忘了,所以才有那么多内存泄漏。
ES6 考虑到这点,推出了两种新的数据结构:weakset 和 weakmap 。他们对值的引用都是不计入垃圾回收机制的,也就是说,如果其他对象都不再引用该对象,那么垃圾回收机制会自动回收该对象所占用的内存。

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

上面代码中,先新建一个 Weakmap 实例。然后,将一个 DOM 节点作为键名存入该实例,并将一些附加信息作为键值,一起存放在 WeakMap 里面。这时,WeakMap 里面对 element 的引用就是弱引用,不会被计入垃圾回收机制。
注册监听事件的 listener 对象很适合用 WeakMap 来实现。

  1. // 代码1
  2. ele.addEventListener('click', handler, false)
  3. // 代码2
  4. const listener = new WeakMap()
  5. listener.set(ele, handler)
  6. ele.addEventListener('click', listener.get(ele), false)

代码 2 比起代码 1 的好处是:由于监听函数是放在 WeakMap 里面,一旦 dom 对象 ele 消失,与它绑定的监听函数 handler 也会自动消失。

发表评论

表情:
评论列表 (有 0 条评论,384人围观)

还没有评论,来说两句吧...

相关阅读