Java中对象的死亡

       Java中是通过可达性分析来判定对象是否存活的。这个算法的思路是通过一系列的称为”GC Roots“的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明这个对象是不可用的。JAVA中可作为GC Roots的对象包括下面几种:1、虚拟机栈中引用的对象  2、方法区中类静态属性引用的对象  3、方法去中常量引用的对象 4、本地方法栈中JNI(本地方法)引用的对象。

        即使在可达性分析算法中不可达的对象,也并非是非死不可得,这时候它们是暂时处于缓刑阶段,要真正宣告一个对象死亡,至少要经历两次标记过程:如果在对象进行可达性分析后发现没有与GC Roots相连接的引用链,那么它将会被第一次标记并且进行一次筛选,筛选的条件是此对象是否有必要执行fianlize()方法。当对象没有覆盖finalize()方法,或者finalize()方法已经被虚拟机调用过,虚拟机将这两种情况都视为没有必要执行。即,如果对象没有覆盖finalize()方法,finalize()方法已经被JVM调用过,那么没有必要执行finalize()方法。(注:fianlize()方法作用:GC确定不存在该对象的更多引用时,由GC调用该对象的fianlize方法)。若判定为有必要执行finalize()方法,那么这个对象将放置在一个叫做F-Queue的队列中,并在稍后由一个由虚拟机自动建立的、低优先级的Finalizer线程去执行它。如下代码:

public class TestFinalize {
	
	public static TestFinalize testFinalize=null;
	
	public void finalize() throws Throwable{
		super.finalize();
		System.out.println("正在执行finalize方法~!");
		testFinalize=this;    //重新与引用链上的对象建立关联
	}
	
	public static void main(String[] args) throws InterruptedException{
		testFinalize=new TestFinalize();  
		
		/*  对象第一次成功拯救自己~!             **/
		testFinalize=null;
		System.gc();
		Thread.sleep(1000); //因为finalize()方法优先级很低,所以暂停1S等待它
		if(testFinalize!=null){         //finalize()方法确实被GC触发了,但是收集前成功逃脱了。
			System.out.println("testFinalize还活着~!");  
		}else{
			System.out.println("testFinalize已经死了~!");
		}
		
		/*  对象第二次成功拯救自己未遂~! 与上面的代码完全一致            **/
		testFinalize=null;
		System.gc();
		Thread.sleep(1000); //因为finalize()方法优先级很低,所以暂停1S等待它
		if(testFinalize!=null){              
			System.out.println("testFinalize还活着~!");
		}else{
			System.out.println("testFinalize已经死了~!");
		}
		
	}
	
	
}

输出结果:

正在执行finalize方法~!
testFinalize还活着~!
testFinalize已经死了~!

上面两段代码完全一样,但是一个成功逃脱被回收,一个失败,因为 任何一个对象的finalize()方法只能被系统自动调用一次。如果对象面临下一次回收,那么它的finalize()方法,不会被执行。

你可能感兴趣的:(深入理解JVM)