转载请注明出处
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只保证会尽最大的努力执行,但是具体什么时候执行以及会不会执行都是未知的。