Java对象死亡标记的过程
在Java可达性标记算法中,要宣告一个对象死亡,至少要进行两次标记过程:
原理摘要
- 首先,如果发现对象从GC root出发不可达,那么就会被第一次标记并进行筛选。筛选条件是此对象是否有必要执行finalize()方法。当对象没有覆盖finalize()方法,或者finalize()方法已经被执行过了,则会被判定为“没有必要执行”。
- 在确定对象有必要执行finalize()方法后,它将会被放入一个队列中,并由一个较低级别的线程去执行此方法。稍后GC会对此队列进行第二次标记,如果发现依旧从GC roots不可达,那么此对象基本上是要被真的回收了。
- 也就是说,finalize()方法是对象逃脱死亡的最后一根稻草。如果对象可以在此方法中将自己建立起与GC roots 的可达性,也就是有其他人引用了此对象,那么就可被免于清理
- 注意:finalize只会被执行一次!
代码验证
class MyObject {
public static MyObject sSaveHook;
@Override
protected void finalize() throws Throwable {
super.finalize();
System.out.println("finalize is executed!");
sSaveHook = this;
}
}
public class Main {
public static void main(String[] args) throws Exception {
MyObject.sSaveHook = new MyObject();
MyObject.sSaveHook = null;
//======第一次GC======
System.gc();
Thread.sleep(500); // 由于finalize方法优先级较低,所有暂停主线程0.5秒以等待它的执行
if(MyObject.sSaveHook != null){
System.out.println("object is alive!"); //这是第一次的输出结果
} else {
System.out.println("object is dead");
}
//======第二次GC======
MyObject.sSaveHook = null;
System.gc();
Thread.sleep(500);
if(MyObject.sSaveHook != null){
System.out.println("object is alive!");
} else {
System.out.println("object is dead"); //这是第二次的输出结果
}
}
}
第一次成功脱险,因为调用finalize方法时这个对象重新找到了引用它的对象。第二次没有,是因为finalize方法只会被执行一次。也就说,第二次GC时没有调用finalize方法,因此该对象对GC roots来说依旧不可达,结果被清除。
注意:由于实验需要,笔者覆写了finalize方法。但由于它的运行大家高昂,不确定性大,因此在实际应用中应避免finalize的使用。