一、分代收集
根据对象存活周期的不同将内存划分为几块,如新生代、老年代、永久代,这样就可以根据各年代特点分别采用最适当的GC算法。
- 新生代采用复制算法
在新生代每次垃圾收集都能发现大批对象已死,只有少量存活。因此选用复制算法,只需要付出少量存活对象的复制成本就可以完成收集。
新生代GC较频繁,但GC效率高。
- 老年代采用标记整理算法
老年代对象存活率高、没有额外空间对它进行分配担保。因此采用“标记-清理”或“标记-整理”算法来进行回收,不必进行内存复制,且直接腾出空闲内存。
老年代GC不频繁,但GC效率不高。
- 永久代(方法区)
在方法区进行垃圾回收一般“性价比”较低,因为在方法区主要回收:废弃常量和无用的类。回收废弃常量与回收其它年代中的对象类似,但要判断一个类是否无用条件相当苛刻:
1、该类所有的实例都已经被回收,Java堆中不存在该类的任何实例。
2、该类对应的Class对象没有在任何地方被引用(也就是在任何地方都没有通过反射访问该类的方法)。
3、加载该类的ClassLoader已经被回收。
但即使满足以上内容也未必一定会回收,HotSpot VM提供了-Xnoclassgc参数来关闭Class的垃圾回收。在大量使用动态代理、CGLib等字节码框架的应用中一定要关闭该选项,开启VM的类卸载功能,以保证方法区不会溢出。
二、分区收集
分区算法将整个堆空间划分为连续的不同小区间,每个小区间独立使用,独立回收。这样做的好处是可以控制一次回收多少个小区间,根据目标停顿时间,每次合理地回收若干个小区间(而不是整个堆),从而减少一次GC所产生的停顿。