垃圾收集器
主要通过阅读《深入了解Java虚拟机》(周志明 著)和网络资源汇集而成,为本人学习JVM的笔记。同时,本文理论基于JDK 1.7版本,暂不考虑 1.8和1.9 的新特性,但可能初略提到。
一、GC概念
垃圾收集(GC,Garbage Collection),就是在动态分配内存后对内存进行自动回收。
- 哪些内存需要回收?
- 已死对象所占的内存需要回收 。
- 什么时候回收?
- 当内存不够用时执行垃圾回收,主要分为 Minor GC(新生代垃圾回收) 和 Major GC(又称 Full GC,老年代垃圾回收)。
- 堆(head)可以分为 Eden Space(新手村)、Survivor Space(幸存者区) 和 Tenured Gen(养老区)。对象会被优先分配到 Eden 区,大对象会直接分配到 Tenured Gen。
- 当 Eden 区满了的话发生 Minor GC,有引用的对象将被移到 Survivor 区。Survivor 区定期(可以自定义)进行GC;经历过一定次数GC仍然幸存的对象,将被送入到 Tenured Gen。当Tenured Gen 满了会发生 Major GC,或者受 HandlePromotionFailure 参数控制强制 Major GC。
- 如何回收?
新生代由于对象产生比较多并且大都是朝生夕灭,所以一般使用复制算法
;而老年代的生命力很强,所以建议使用标记-助理算法
。
二、判断对象是否可以回收
程序计数器、虚拟机栈和本地方法栈,对于线程而言是私有的。当线程结束时,它们会被自动回收,所以不需要过多考虑回收问题。
对于Java程序而言,需要回收内存的地方主要就是堆(大部分对象的存放位置),回收对堆内存的分配。判断方法:
1. 引用计数法
给对象添加一个引用计数器,每当有一个地方引用它时,计数器值加1;当引用失效时,计数器 值减1;如何时刻计数器值为0的对象就是不能再被使用的。但是,这种方法难以解决对象之间相互循环引用的问题(主流的Java虚拟机没有引用这种计数方法来管理内存)。
2. 可达性分析算法(根搜索算法)
通过一系列名为”GC Roots“对象作为起始点,从这些节点开始往下搜索,搜索过的路径称为引用链。当一个对象到 GC Roots 没有任何引用链相连时,则证明该对象是不可用的,会被判定为可回收对象。
GC Roots 对象
(1) 虚拟机栈(栈帧中的本地变量表)中引用的对象。
(2) 方法区中类静态属性引用的对象。
(3) 方法区中常量引用的对象。
(4) 本地方法栈中引用的对象(Native 对象)。
3. 四种引用
以上谈到的两种算法其实都与”引用”有关。
(1)强引用(Strong Reference)
在代码中普遍存在。只要强引用还存在,垃圾收集器永远不会回收被引用的对象。当内存空间不足时,Java虚拟机宁愿抛出OOM,使程序异常终止。
(2)软引用(Soft Reference)
在内存空间不足的情况下,虚拟机才会回收这种对象。软引用只要通过SoftReference 类来实现,可以作为内存敏感的高速缓存。
(3)弱引用(Weak Reference)
GC后都会回收的一类对象,可以通过WeakReference 类来实现。
(4)虚引用(Phantom Reference)
可以通过PhantomReference 类来实现,为一个对象设置虚引用的唯一目的就是能在这个对象收集器回收时收到一个系统通知。
三、生存还是死亡
一个对象死亡的经历:
- 如果对象在进行可行性分析后发现没有与 GC Roots 相连接的引用链,则将其第一次标记并进行一次筛选。
- 筛选条件:该对象有没有必要执行 finalize() 方法
- 没有必要执行的情况:
对象没有覆盖 finalize() 方法;
finalize() 方法已经被虚拟机调用过了。
- 若有必要执行 finalize() 方法,将对象放到 F-Queue 队列中。
- 虚拟机自动建立 Finalizer 线程(低优先级)去执行该方法(PS:若对象在该方法中执行缓慢甚至死循环,会导致严重后果,甚至导致内存回收系统崩溃)。
- finalize() 也是对象逃脱死亡的最后一次机会:
- GC 会对队列中对象进行第二次标记。
- 若对象与引用链上任何一个对象建立关联,即可脱离被回收的命运。
- 第二次标记后,将对象移出队列,并最终被系统回收。
任何一个对象的 finalize() 方法只会被系统调用一次!
finalize() 方法的代价高昂,不确定性大,无法保证各个对象的调用顺序。完全可以使用 try-finally 等方法来实现它的功能。但还是要了解一下Java的对于判断对象存亡的执行机制。
其他
- 当GC与非GC时间耗时超过了 GCTimeRatio 的限制时,会触发 OOM。
- GC调优:
- 用 NewRatio 来控制新生代和老年代的比例。
- 用 MaxTenuringThreshold 来控制进入老年代前的生存次数。
- 使老年代存储空间延迟达到 Major GC。
参考资料
- 《深入了解Java虚拟机》(周志明 著)