什么是内存泄漏,什么情况下会产生?
因为一些代码操作导致一些需要回收的内容不符合回收机制,导致没有被回收!
怎么回收?js的垃圾回收
JavaScript垃圾回收的机制很简单:找出不再使用的变量,然后释放掉其占用的内存,但是这个过程不是实时的,因为其开销比较大,所以垃圾回收系统(GC)会按照固定的时间间隔,周期性的执行。
回收机制
-
全局变量不会被回收。
-
局部变量会被回收,也就是函数一旦运行完以后,函数内部的东西都会被销毁。
-
只要被另外一个作用域所引用就不会被回收。
回收方法
- 标记清除,在局部变量创建时,会被标记为进入环境标记,这个时候是不会回收的,当函数执行完毕之后,标记会被标记为离开环境,这个时候就会被清除。
- 引用计数,在一个变量被创建时,每次被引用就会标记引用次数+1,一旦不再使用时引用次数-1,当检测到引用次数为0时,清除掉。
哪些情况会造成内存泄漏?
- 1.意外的全局变量: 由于使用未声明的变量,而意外的创建了一个全局变量,而使这个变量一直留在内存中无法被回收。
-
原因:由于没有写var,在非严格模式下会被认为是全局变量,即window.a===10//true ,全局变量不会被回收。a = 10
-
解决办法:使用严格模式(use strict)
- 2.被遗忘的计时器或回调函数: 设置了 setInterval 定时器,而忘记取消它,如果循环函数有对外部变量的引用的话,那么这个变量会被一直留在内存中,而无法被回收。
- 这个不用举例子把,别忘了及时清除定时器
- 3.脱离 DOM 的引用: 获取一个 DOM 元素的引用,而后面这个元素被删除,由于一直保留了对这个元素的引用,所以它也无法被回收。
-
let gogo=document.getElementByid('go') document.body.removeChild(document.getElementByid('go'))
- 解决办法:删除dom的引用
- 4.闭包: 不合理的使用闭包,从而导致某些变量一直被留在内存当中。
-
function a(){ let i=10 return function(){ console.log(i) } } let b=a()
-
原因:闭包实际上就是在外部函数读取函数内部的变量,这就导致内部变量被其他作用引用,无法回收
-
解决:解除闭包,比如立即执行函数等