关于对象的存活状态的判定可以使用两种算法。
第一种是引用计数算法,每一个对象都会有一个私有计数器,每当有一个地方引用他时,便会+1;相应的,有引用失效时,计数器就会-1,而当计数器变为0时,这个对象就是不可能再被使用的了。
第二种是可达性分析算法,从一个顶点对象(GC Roots)从上向下搜索,走过的路程被叫做引用链,,当有对象和GC Roots之间不可达时,这个或者这些对象就是不可用的了,即使这些对象有许多都是在被关联在一起。
下面是对象的死亡过程。
可以先看一段代码。
public class FinalizeEscapeGC {
static FinalizeEscapeGC SAVE_HOOK=null;
void islive(){
System.out.println("yes,i am still alive :)");
}
@Override
protected void finalize() throws Throwable {
super.finalize();
System.out.println("finalize method executed");
FinalizeEscapeGC.SAVE_HOOK=this;
}
public static void main(String[] args) throws InterruptedException {
SAVE_HOOK=new FinalizeEscapeGC();
SAVE_HOOK=null;
System.gc();
Thread.sleep(500);
if(SAVE_HOOK!=null){
SAVE_HOOK.islive();
}else{
System.out.println("no ,i am dead :(");
}
SAVE_HOOK=null;
System.gc();
Thread.sleep(500);
if(SAVE_HOOK!=null){
SAVE_HOOK.islive();
}else{
System.out.println("no ,i am dead :(");
}
}
}
这是运行结果:
finalize method executed
yes,i am still alive :)
no ,i am dead :(
可以看到finalize()方法制只被用了一次,而且其中第一次回收的时候里面重新赋予了值,,而第二次会直接被回收掉。
在上面的两种算法中判定对象不可存活,当时对象真正的死亡也是要经历两次标记。用可达性分析算法来说,当发现对象没有和GC Roots相连接的可用链时,就会进行一次标记,并进行筛选,假如对象没有覆盖finalize()方法或者已经执行过finalize()方法了,就没有必要执行finalize()方法。
假如没有执行这个方法,它就会被放进一个队列中去,稍等一会GC会对这个队列进行小规模的二次标记,假如它完成了自救,就不用死亡,也就是重新和一个引用链的对象进行连接。没有的话,就会真的被回收掉。
这样一来就可以解释了上面的代码,在第一次被回收的时候进行了标记,还没有被真正的回收掉,也重新和一个引用链上的对象完成了链接,但是标记还在,之后又被赋予了空值,再次被回收,且只有第一次回收进行了finalize()方法,所以第二次回收的时候是被真正的回收掉了。
参考书籍:《深入理解JAVA虚拟机》