在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
为什么需要垃圾回收(GC)
什么是垃圾回收垃圾回收机制也称 垃圾产生当一个对象没有任何的变量或属性对它进行引用,此时我们将永远无法操作该对象,这种对象就是一个垃圾,这种对象过多会占用大量的内存空间,导致程序变慢。 例如: 这里我先声明了一个Person变量,它引用了对象{name: "江流",age: 20},接着我又将这个Person变量指向了另一个对象{name: "心猿", age: 5000},那么之前被引用的对象,现在就成了无用对象,也永远无法使用操作该对象,这种对象就是一个垃圾。 这种垃圾对象过多,就会占用大量空间,如果一直不释放就会影响系统性能,重则导致程序崩溃,所以就需要垃圾回收释放这部分内存。 这个过程我们不需要也不能进行垃圾回收的操作。 我们只需要的是将不再使用的对象设置为null即可。 垃圾回收策略JavaScript 中主要的内存管理概念是可达性。大概意思是以某种方式可以访问到或者可以使用的值,它们就是需要保存在内存中,无法访问,也无法使用的值,则需要被垃圾回收机制回收。 垃圾回收过程是不实时进行的,因为JavaScript是一门单线程的语言,每次执行垃圾回收,会使程序应用逻辑暂停,执行完垃圾后回收在执行应用逻辑,这种行为称为全停顿,所以一般垃圾回收会在cpu闲时进行。 如何通过某种方式找到所谓的垃圾,是垃圾回收的重点,所以下面常见的算法策略,不过这里只说前两种:
引用计数标记策略思想:
例如: let a = { name: "江流", age: 20 }; //此时该对象的引用计数标记为1(a 引用) let b = a; //此时对象的引用计数标记为2(a、b 引用) a = null; //此时对象的引用计数标记为1((b 引用)) b = null; //此时对象的引用计数标记为0(无变量引用) ... //等待GC 回收此对象 但是这种方式有个很严重的问题 – 循环引用 循环引用引发的问题在一个函数中,对象A的属性指向对象B,对象B的属性指向对象A,这个函数在执行完,对象A和B的计数器也不会为0,影响了正常的GC。 例如下面的例子: function test() { let A = new Object(); let B = new Object(); A.pointer= B; B.pointer = A; } test(); 当对象A和对象B的属性相互引用这,按照引用计数策略,他们的引用计数都是为2,但是在test()执行完成后,在函数执行完,函数作用域中的数据对象A和对象B都应该被GC销毁掉。 如果执行多次,将会造成严重的内存泄漏。 解决方法在函数结束时,将其指向null //切断引用关系 A = null; B = null; 引用计数算法的优缺点优点:
缺点:
标记清除算法核心思想分标记和清除两个阶段完成。 大概过程:
标记清除算法优缺点优点:
缺点: 在清除垃圾之后,剩余对象的内存位置是不变的,就会导致空闲内存空间不连续。这样就出现了内存碎片,并且由于剩余空间不是整块,就需要内存分配的问题。 标记整理算法标记整理(Mark-Compact)算法,就是可以有效的解决,它是在标记结束后标记整理算法会将不需要清理的对象向内存一端移动,最后清理边界的内存。 V8引擎的垃圾回收
针对不同对象采用不同算法: (1)新生代:对象的存活时间较短。新生对象或只经过一次垃圾回收的对象。 (2)老生代:对象存活时间较长。经历过一次或多次垃圾回收的对象。 回收新生代对象回收新生代对象主要采用复制算法(Scavenge 算法)加标记整理算法。而Scavenge 算法的具体实现,主要采用了Cheney算法。 对象晋升机制一轮GC还存活的新生代需要晋升。 回收老生代对象回收老生代对象主要采用标记清除、标记整理、增量标记算法,主要使用标记清除算法,只有在内存分配不足时,采用标记整理算法。
参考文档:总结
|
请发表评论