内存泄漏是前端开发中的一个常见问题,可能导致项目变得缓慢、不稳定甚至崩溃。内存泄漏是指不再用到的内存没有及时被释放,从而造成内存上的浪费。
JavaScript 垃圾回收机制很简单:找出不再使用的变量,然后释放掉其占用的内存。
① 标记-清除
对内存中所有变量加上标记,然后再去掉进入环境的变量的标记,最后就是清除那些带标记的变量并回收它们所占用的内存空间。
当变量进入环境时,就将这个变量标记为“进入环境”。
当变量离开环境时,这将其标记为“离开环境”。
② 引用计数
语言引擎有一张”引用表”,跟踪记录每个值被引用的次数。
如果一个值的引用次数是0,就表示这个值不再用到了,因此可以将这块内存释放。
出现内存泄漏的情况:
会持有对DOM的引用,妨碍垃圾回收器释放相关的内存。
例如使用了addEventListener,那就要记得 removeEventListener
当两个或多个对象相互引用时,即使你不再使用它们,它们也无法被垃圾回收。
解决方法:确保在不再需要对象时,将其引用设置为null,打破循环引用。
原因:定时器 setInterval 或者 setTimeout 在不需要使用的时候,没有被clear,导致定时器的回调函数及其内部依赖的变量都不能被回收,这就会造成内存泄漏。
解决方法:在不再需要定时器或间隔器时,使用 clearTimeout 和 clearInterval 来清理它们。
利用谷歌浏览器调试工具 Memory 的内存快照 Heap snapshot,找到使你内存增加的业务场景。
虽然浏览器可以进行垃圾自动回收,但是当代码比较复杂时,垃圾回收所带来的代价较大,所以应该尽量减少垃圾回收。
在清空一个数组时,最简单的方法就是给其赋值为[ ],但是与此同时会创建一个新的空对象,可以将数组的长度设置为0,以此来达到清空数组的目的。
对象尽量复用,对于不再使用的对象,就将其设置为null,尽快被回收
在循环中的函数表达式,如果可以复用,尽量放在函数的外面。
其它:
Vue组件触发 beforeDestroy 声明钩子函数的时候,把变量置null(那些用来渲染页面的数据量大的Object、Array等;也可以全部置 null ;包括dom绑定的事件要移除、定时器要清除)
组件中使用 v-show 可能会导致 DOM 元素积累,而销毁 DOM 的核心其实还是使用v-if,尽量使用 v-if 代替v-show,同时避免不必要的DOM操作。