垃圾回收关注的是堆和方法区两个区域。
可达性分析算法
以“GC Roots”对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots没有任何引用链相连,则此对象是不可用的。
可做为GC Roots的对象包括以下几种:
- 虚拟机栈(栈帧中的本地变量表)中引用的对象
- 本地方法栈中JNI(native方法)引用的对象
- 方法区中类静态属性引用的对象
- 方法区中常量引用的对象
引用类型
- 强引用(Strong Reference):通过类似Object obj=new Object()创建出来的引用
- 软引用(Soft Reference):在将要发生内存溢出异常之前,会将这些对象进行回收,如果回收之后还是没有足够的内存,则会抛出内存溢出异常。使用SoftReference实现。
- 弱引用(Weak Reference):当垃圾收集器工作时,无论当前内存是否足够,都会回收掉只被弱引用关联的对象。使用WeakReference类实现。
- 虚引用(Phantom Reference):在回收该对象时会受到一个系统通知。使用PhantomReference实现。
确定堆中需要被回收的对象
堆中的一个对象被回收之前需要通过两个关卡:
- 在可达性分析算法中不可达
- finalize方法已被执行(在finalize方法中可以把对象设置为可达对象,这样可以使对象“重生”,但是在对象下一次不可达的时候,finalize不会再次被执行,也就是说finalize最多只被执行一次)。
finalize不是一定会被执行,各个对象执行的时间点和顺序也很不确定,所以不建议使用它。
方法区
方法区(永久代)的垃圾收集主要回收两部分内容:废弃常量和无用的类。
DirectMemory
32位Windows平台限制每个进程最大可用内存为2GB,“DirectMemory+堆+栈+其他内存”不能大于这个限制。DirectMemory不能像新生代、老年代那样,发现空间不足了就通知收集器进行垃圾回收,它只能等待老年代满了后Full GC,然后“顺便地”帮它清理掉内存的废弃对象。否则它只能一直等到抛出内存溢出异常时,先catch掉,再在catch块里面调用System.gc()。
补充说明
在可达性分析搜索根节点(GC Roots)时,GC进行时必须停顿所有的线程(Stop The World),以确保对象引用关系停止变化。
线程需要在安全点或安全区域中断,一般采用主动式中断。