finalize()和System.gc()的区别

时间:2021-11-18 03:38:34

finalize()是由JVM自动调用的,你可以用System.gc(),但JVM不一定会立刻执行,JVM感觉内存空间有限时,才会开始执行finalize(),至于新的对象创建个数和被收集个数不同是因为收集的对象只和JVM的垃圾收集策略有关。

(1) finalize()函数是干嘛的?Java不是有Garbage Collection(以下简称gc)来负责回收内存吗?
gc只能清除在堆上分配的内存(纯java语言的所有对象都在堆上使用new分配内存),而不能清除栈上分配的内存(当使用JNI技术时,可能会在栈上分配内存,例如java调用c程序,而该c程序使用malloc分配内存时)。因此,如果某些对象被分配了栈上的内存区域,那gc就管不着了,对这样的对象进行内存回收就要靠finalize()。
举个例子来说,当java调用非java方法时(这种方法可能是c或是c++的),在非java代码内部也许调用了c的malloc()函数来分配内存,而且除非调用那个了free()否则不会释放内存(因为free()是c的函数),这个时候要进行释放内存的工作,gc是不起作用的,因而需要在finalize()内部的一个固有方法调用它(free())。
finalize的工作原理应该是这样的:一旦垃圾收集器准备好释放对象占用的存储空间,它首先调用finalize(),而且只有在下一次垃圾收集过程中,才会真正回收对象的内存。所以如果使用finalize(),就可以在垃圾收集期间进行一些重要的清除或清扫工作。

(2) finalize()在什么时候被调用?
有三种情况
1.所有对象被Garbage Collection时自动调用,比如运行System.gc()的时候;
2.程序退出时为每个对象调用一次finalize方法;
3.显式的调用finalize方法
除此以外,正常情况下,当某个对象被系统收集为无用信息的时候,finalize()将被自动调用,但是jvm不保证finalize()一定被调用,也就是说,finalize()的调用是不确定的,这也就是为什么sun不提倡使用finalize()的原因。

[java] view plaincopy
  1. //: initialization/E10_FinalizeCall.java  
  2. /****************** Exercise 10 ***************** 
  3.  * Create a class with a finalize() method that 
  4.  * prints a message. In main(), create an object 
  5.  * of your class. Explain the behavior of your 
  6.  * program. 
  7.  ************************************************/  
  8. package initialization;  
  9.   
  10. public class E10_FinalizeCall {  
  11.     protected void finalize() {  
  12.         System.out.println("finalize() called");  
  13.     }  
  14.   
  15.     public static void main(String args[]) {  
  16.         new E10_FinalizeCall();  
  17.     }  
  18. ///:~  

 

你可能发现finalizer没有被调用,因为程序经常没有产生很多的垃圾来触发收集器。

[java] view plaincopy
  1. //: initialization/E11_FinalizeAlwaysCalled.java  
  2. /****************** Exercise 11 **************** 
  3.  * Modify Exercise 10 so your 
  4.  * finalize() will always be called. 
  5.  ***********************************************/  
  6. package initialization;  
  7.   
  8. public class E11_FinalizeAlwaysCalled {  
  9.     protected void finalize() {  
  10.         System.out.println("finalize() called");  
  11.     }  
  12.   
  13.     public static void main(String args[]) {  
  14.         new E11_FinalizeAlwaysCalled();  
  15.         System.gc();  
  16.         System.runFinalization();  
  17.     }  
  18. /* 
  19. * Output: finalize() called 
  20. *///:~  

连续调用System.gc()和System.runFinalization()可能但是不一定会调用你的Finalizer(finalize是否会运行根据不同的JDK版本而不确定)。调用这些方法只是一个请求,它不代表finalizer一定会运行。总之,没有什么可以保证finalize()方法一定会被调用。