看这个贴子,这位兄弟说他在finalize方法里面执行了一些db connnection的操作,可能是关闭connection的事情,结果引起了OutOfMemoryError的错误.
他告诉我们千万别在finalizer里面进行I/O操作.我要补充的是,不要在finalize里面做些可能长时间才能完成的操作.
因为它很容易让GC挂起的.看java.lang.ref.Finalizer类你就知道,一个对象里面的finalize()方法是由JVM来调用的,多个对象的finalize的执行顺序就是一个队列,要一个一个的执行,执行过程中GC会暂停,如果finalize执行时间太长,那是一件非常危险的事情.
我写了一段代码来模拟这个错误
public class FinalizerTest {
private volatile static int invokeCount;
private int objectId;
private byte[] bytes;
public FinalizerTest(int objectId) {
this.objectId = objectId;
}
public void newBytes() {
bytes = new byte[objectId];
}
public void finalize() throws InterruptedException{
System.out.println("InvokeCount." + (invokeCount++) + ".Object." + objectId + ".Thread." + Thread.currentThread().getId() + ".finalize()");
Thread.sleep(100);//模拟长时间操作
}
public static void main(String[] args) throws InterruptedException {
int count = 100000;
for(int i=0;i<count;i++){
FinalizerTest ft = new FinalizerTest(i);
ft.newBytes();
}
}
}
你可以看到main函数里面的for循环操作就是代表快速的内存空间分配和垃圾产生的过程.如果收集这些垃圾不能及时完成,那么所有的空间很快就会用完,OutOfMemoryError就很快出现了.
从这点还能看到一个问题,java这种虚拟机接手所有的垃圾回收工作的语言,在使用的时候还是要特别小心.