前言
Java 不像 C 一样,需要程序员手动管理内存,JVM 会自动进行垃圾回收。垃圾回收是对堆和方法区进行垃圾回收,因为这两个区域是线程公有;线程私有的数据,不用进行垃圾回收(如:栈)。
JVM 是如何完成对对象的判断,然后进行垃圾回收的呢?早期使用的是引用计数法
,现在使用的一般都是可达性分析
。
1.引用计数法
给对象添加一个引用计数器,每当有一个地方引用的时候,计数器的值就 +1。当引用失效的时候,计数器的值就 -1。任何时刻计数器为0的对象就是不可能再被引用的
。
优点:简单、高效
问题:难以解决对象之间循环引用的问题。
public class Main {
public static void main(String[] args) {
DataObject d1 = new DataObject();
DataObject d2 = new DataObject();
d1.o = d2;
d2.o = d1;
d1 = null;
d2 = null;
}
}
class DataObject{
public Object o = null;
}
如上代码所示,d1 的引用计数为1,d2 的引用计数也为1,当把对象 d1 和 d2 置为 null 之后,d1、d2之间的引用还是存在的。也就导致了 d1 和 d2 不能被GC回收。
2.可达性分析
通过一系列的称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链(Reference Chain)
,当一个对象到GC Roots没有任何引用链相连时,则证明此对象是不可用的。
在Java语言中,可作为GC Roots的对象包括下面几种:
- 虚拟机栈(栈帧中的本地变量表)中引用的对象。
- 本地方法栈中JNI (即一般说的Native方法)引用的对象。
- 方法区中类静态属性引用的对象。
- 方法区中常量引用的对象。