JVM垃圾处理方法(标记清除、复制、标记整理)
- 标记-清除算法
标记阶段:先通过根节点,标记所有从根节点开始的对象,未被标记的为垃圾对象
清除阶段:清除所有未被标记的对象
- 复制算法
将原有的内存空间分为两块,每次只使用其中一块,在垃圾回收时,将正在使用的内存中的存活对象复制到未使用的内存块中,然后清除正在使用的内存块中的所有对象。
- 标记-整理
标记阶段:先通过根节点,标记所有从根节点开始的可达对象,为被标记的为垃圾对象
整理阶段:将所有的存活对象压缩到内存的一段,之后清理边界所有的空间
三种算法的比较
-
效率:复制算法 > 标记/整理算法 > 标记/清除算法(此处的效率只是简单的对比时间复杂度,实际情况不一定如此)。
-
内存整齐度:复制算法=标记/整理算法>标记/清除算法。
-
内存利用率:标记/整理算法=标记/清除算法>复制算法。
JVM如何GC,新生代,老年代,持久代,都存储哪些东西,以及各个区的作用?
新生代
- 在方法中去new一个对象,那这方法调用完毕后,对象就会被回收,这就是一个典型的新生代对象。
老年代
- 在新生代中经历了N次垃圾回收后仍然存活的对象就会被放到老年代中。而且大对象直接进入老年代
当Survivor空间不够用时,需要依赖于老年代进行分配担保,所以大对象直接进入老年代
永久代
- 即方法区。
GC用的引用可达性分析算法中,哪些对象可作为GC Roots对象?
-
Java虚拟机栈中的对象
-
方法区中的静态成员
-
方法区中的常量引用对象
-
本地方法区中的JNI(Native方法)引用对象。
什么时候进行MinGC,FullGC
MinGC
-
新生代中的垃圾收集动作,采用的是复制算法
-
对于较大的对象,在Minor GC的时候可以直接进入老年代
FullGC
-
Full GC是发生在老年代的垃圾收集动作,采用的是标记-清除/整理算法。
-
由于老年代的对象几乎都是在Survivor区熬过来的,不会那么容易死掉。因此Full GC发生的次数不会有MinorGC那么频繁,并且Time(Full GC)>Time(Minor GC)
各个垃圾收集器是怎么工作的
Serial收集器
- 是一个单线程的收集器,不是只能使用一个CPU。在进行垃圾收集时,必须暂停其他所有的工作线程,直到收集结束。
- 新生代采用复制算法,Stop-The-World
- 老年代采用标记-整理算法,Stop-The-World
- 简单高效,Client模式下默认的新生代收集器
ParNew收集器
- ParNew收集器是Serial收集器的多线程版本
- 新生代采用复制算法,Stop-The-World
- 老年代采用标记-整理算法,Stop-The-World
- 它是运行在Server模式下首选新生代收集器
- 除了Serial收集器之外,只有它能和CMS收集器配合工作
ParNew Scanvenge收集器
-
类似ParNew,但更加关注吞吐量。目标是:达到一个可控制吞吐量的收集器。
-
停顿时间和吞吐量不可能同时调优。我们一方面希望停顿时间少,另外一方面希望吞吐量高,其实这是矛盾的。因为:在GC的时候,垃圾回收的工作总量是不变的,如果将停顿时间减少,那频率就会提高;既然频率提高了,说明就会频繁的进行GC,那吞吐量就会减少,性能就会降低。
G1收集器
- 是当今收集器发展的最前言成果之一,对垃圾回收进行了划分优先级的操作,这种有优先级的区域回收方式保证了它的高效率
- 最大的优点是结合了空间整合,不会产生大量的碎片,也降低了进行gc的频率
- 让使用者明确指定指定停顿时间
CMS收集器:(Concurrent Mark Sweep:并发标记清除老年代收集器)
-
一种以获得最短回收停顿时间为目标的收集器,适用于互联网站或者B/S系统的服务器上
-
初始标记(Stop-The-World):根可以直接关联到的对象
-
并发标记(和用户线程一起):主要标记过程,标记全部对象
-
重新标记(Stop-The-World):由于并发标记时,用户线程依然运行,因此在正式清理前,再做修正
-
并发清除(和用户线程一起):基于标记结果,直接清理对象
-
并发收集,低停顿