随时随地阅读更多技术实战干货,获取项目源码、学习资料,请关注源代码社区公众号(ydmsq666)
当垃圾回收机制回收某个对象所占用的内存之前,通常要求程序调用适当的方法来清理资源,在没有明确指定资源清理的情况下,Java提供了默认机制来清理资源,即调用finalize方法,它是Object类的方法,原型为:protected void finalize() throws Throwable ,任何Java类都可以覆盖Object类的finalize方法。 它可以抛出任何类型的异常。垃圾回收机制何时调用对象的finalize方法是完全透明的,只有当程序需要更多额外内存时,才会进行垃圾回收。
finalize方法特点:
1、永远不要主动调用某个对象的finalize方法,该方法应该交给垃圾回收机制调用。
2、该方法何时被调用,是否被调用具有不确定性。
3、当JVM执行去活对象的finalize方法时,可能使该对象或系统中其他对象重新变为激活状态。
4、当JVM 执行finalize方法时出现了异常,垃圾回收机制不会报告异常,程序继续执行。
下面示例演示了如何在finalize方法里把自身复合,通过该程序可以看出垃圾回收的不确定性。
package com.home.finalize;
public class TestFinalize {
public static TestFinalize tf = null;
public void info() {
System.out.println("测试finalize方法");
}
public static void main(String[] args) throws Exception {
// 创建TestFinalize对象立即进入去活状态
new TestFinalize();
// 通知系统进行资源回收
System.gc();
// 让程序暂停2秒
Thread.sleep(2000);
tf.info();
}
public void finalize() {
tf = this;
}
}
该程序重写了finalize方法,把需要清理的去活对象重写赋给了tf引用变量,从而该对象重新被激活。
另外,如果程序中去掉休眠代码,会报空指针,这说明当调用System.gc()方法后,系统并未立即进行垃圾回收。
另外,System和Runtime类有一个runFinalization方法可以强制垃圾回收机制调用系统中去活对象的finalize方法,就不必使用休眠了,代码如下:
package com.home.finalize;
public class TestFinalize {
public static TestFinalize tf = null;
public void info() {
System.out.println("测试finalize方法");
}
public static void main(String[] args) throws Exception {
new TestFinalize();// 创建TestFinalize对象立即进入去活状态
System.gc();// 通知系统进行资源回收
Runtime.getRuntime().runFinalization();// 强制垃圾回收机制调用去活对象的finalize方法
// System.runFinalization();//与上一句代码效果一样
tf.info();
}
public void finalize() {
tf = this;
}
}
}
经过测试,程序正常运行。
注:本文知识点内容及示例来源于疯狂JAVA讲义一书。