垃圾回收机制与内存泄漏

垃圾回收机制

找出不再使用的变量,然后释放掉其占用的内存,但是这个过程不是实时的,因为其开销比较大,所以垃圾回收系统(GC)会按照固定的时间间隔,周期性的执行。

各浏览器常用的垃圾回收机制:1.引用计数 2.标记清除

引用计数

跟踪记录每个值被引用的次数。当声明了一个变量并将一个引用类型指向该变量时,则这个值的引用次数就是1。那么,当引用指向了别的值,则这个值的引用次数就减1。当这个引用次数变成0时,说明没有办法再访问这个值了,因而就可以将其所占的内存空间给收回来。这样,垃圾收集器下次再运行时,它就会释放那些引用次数为0的值所占的内存。
但这种方式存在一个问题:

var a = {}
var b = {}
a.v = b
b.v = a

以上的循环引用是没办法被垃圾回收掉的,所以就会造成永久的内存泄漏。这就是引用计数的缺点,所以这种方式并不是主流。

标记清除

最常用的垃圾回收方式。
当变量进入环境(如函数中声明一个变量)时,就将这个变量标记为“进入环境”。从逻辑上讲,永远不能释放“进入环境”的变量所占的内存。当变量离开环境时,则将其标记为“离开环境”。
垃圾回收在运行时,会给所有变量都加上另一种标记,比如叫做C标记(并不是上面所讲的“进入环境”跟“离开环境”这两种标记),之后会清除掉“进入环境”的变量的C标记以及他们所引用的变量的C标记。那么剩下的带有C标记的变量就是没有被使用的,此时并不会立刻回收他们的内存,而是在下次垃圾回收启动时被重复加上了标y忆,就会被视为准备删除的变量。最后,垃圾回收器进行内存清理,销毁那些被视为准备删除的变量。

内存泄漏

指一块被分配的内存既不能使用,又不能回收,一直占着内存直到浏览器进程结束。

常见内存泄漏
  1. 意外的全局变量
function setValue() {
	value = "888"
}
setValue();
// 如果这是在全局作用域下调用,则value会变成window的变量,就不会被回收
  1. setInterval()定时器未清除
  2. 闭包
  3. 注意dom的操作。比如某个变量保留了dom元素的引用,当该元素被删除后,因为被变量所引用,所以该元素的dom对象依然保持存在着不会被回收掉,就造成了内存泄漏。
dom引用造成的内存泄漏

这里写图片描述
垃圾回收机制与内存泄漏_第1张图片
垃圾回收机制与内存泄漏_第2张图片
垃圾回收机制与内存泄漏_第3张图片垃圾回收机制与内存泄漏_第4张图片
以上现象很容易理解。实际document.body.removeChild(refA)是能够移除成功的,界面也会改变。但refA.children间接有着refB,refB.parentNode间接有着refA,所以只有当这两个对象都同时没有被变量所指向的时候,在内存中才会被释放。

你可能感兴趣的:(JavaScript)