java之finalization与GC

转载请注明出处

http://blog.csdn.net/pony_maggie/article/details/43925743


作者:小马

 

java 中的GC(garbage collection)并不像C++的析构函数一样每次对象被删除时调用,事实上它是一种JVM(JAVA 虚拟机)行为,JVM在它认为“合适”的情况下调用。这个所谓的合适情况,应用层其实不必考虑过多。可以认为GC调用的时机是随机的。

 

finaliztion是个可以被应用层重载的函数,当GC回收一个对象时,它会先调用对象的这个函数然后再处理其它事情。所以应用可以重载这个函数,去做一下想在GC前收尾的工作(听起来越来越像C++中的析构了,但是请注意调用时机)。

 

但是问题来了,GC本来就是内存回收了,应用还需要在finalization做什么呢? 答案是大部分时候,什么都不用做(也就是不需要重载)。只有在某些很特殊的情况下,比如你调用了一些native的方法(一般是C写的),可以要在finaliztion里去调用C的释放函数。

 

下面这个代码示例,重载了finaliztion,但是仅仅是为了观察GC的行为,类很简单,

class Chair
{
	static boolean gcrun = false;
	static boolean f = false;
	static int created = 0;
	static int finalized = 0;
	
	int i;
	Chair() 
	{
		i = ++created;
		if(created == 47)
		{
			System.out.println("Created 47");
		}
	}
	
	public void finalize()
	{
		if(!gcrun)
		{
			gcrun = true;
			System.out.println("beginning to finalize after " +
					created + " chairs have been created");
		}
		if(i == 47)
		{
			System.out.println("finalizing chair #47, " +
					"setting flag to stop chair creation");
			f = true;
		}
		finalized++;
		if(finalized >= created)
		{
			System.out.println("all " + finalized + " finalized");
		}
	}
}

created和finalized用来跟踪分别创建和回收了多少对象,而对象的成员i可以理解为每个对象的号码,从代码上看,当号码为47的对象被回收时,把f置为true,这样main函数就不会创建新的Chair实例了,main函数如下:

public class Garbage 
{

	/**
	 * @param args
	 */
	public static void main(String[] args) 
	{
		while(!Chair.f)
		{
			new Chair();
			new String("to take up space");
		}
		
		System.out.println("after all chairs have been created:\n" +
				"total finalized = " + Chair.finalized);
		
		//optional arguments force garbage
		//collection & finalization
		if(args.length > 0)
		{
			if(args[0].equals("gc") ||
					args[0].equals("all"))
			{
				System.out.println("gc():");
				System.gc();
			}
			
			if(args[0].equals("finalize") ||
					args[0].equals("all"))
			{
				System.out.println("run finalization():");
				System.runFinalization();
			}
		}
		
		System.out.println("byte!");

	}

}


注意并不是说,创建了第47个对象时就开始回收,原因还是上面讲的回收时机问题。所以当i==47时,可以已经创建了几千个chair对象了。

 

输出结果:

Created 47
beginning to finalize after 4691 chairs have been created
finalizing chair #47, setting flag to stop chair creation
after all chairs have been created:
total finalized = 6151
byte!


main函数中的System.runFinalization()和System.gc是做什么的呢? 我个人的理解,这两个函数分别是应用层向JVM发出一个信号,告诉JVM,希望你能尽快的回收内存和调用对象的finaliztion方法,但是只是一个请求,而JVM只保证会尽最大的努力执行,但是具体什么时候执行以及会不会执行都是未知的。

你可能感兴趣的:(java之finalization与GC)