终结方法finalize()通常是不可预测的,也是很危险的,一般情况下是不必要的。
- Java语言规范并不保证finalize()会被及时执行,即不确定终结方法执行时间,只规定在对象被垃圾回收之前执行
- 不应该依赖终结方法来改变重要的持久状态
- 终结方法会造成严重的性能损耗
正确使用终结方法
防止用户使用了创建了对象后,并未使对象提供的显式终止方法(如果有)。终结方法可以充当安全网。
本地对等体(是一个本地对象,普通对象通过本地方法委托给一个本地对象。)不是一个普通对象,垃圾回收器不会回收它。一般情况下,终结方法需要能够完成所有必要的工作释放资源,如果需要即时释放资源,那么就需要给本地对等体指定一个显式的终止方法。
终结方法链不会自动执行,如果类有终结方法,并且子类覆盖了终结方法,则子类的终结方法必须手工调用超类的终结方法。你应该在try块中终结子类,并在相应的finally块中调用超类的终结方法。
终结方法守卫者
如果子类实现者覆盖了超类的终结方法,但是忘了调用超类的终结方法,那么超类的终结方法永远不会调用。为了防止此种情况出现,可以使用终结方法守卫者。即为每个被终结的对象创建一个附加的对象,该附加对象为一个匿名类,将外围类的终结操作如释放资源放入该匿名类的终结方法中。同时,外围类保存着对该匿名类的唯一引用,即复制给私有变量域。
class A {
@SuppressWarnings("unused")
//终结守卫者
private final Object finalizerGuardian = new Object() {
@Override
//终结守卫者的终结方法将被执行
protected void finalize() {
System.out.println("A finalize by the finalizerGuardian");
}
};
@Override
//由于终结方法被子类覆盖,该终结方法并不会被执行
protected void finalize() {
System.out.println("A finalize by the finalize method");
}
public static void main(String[] args) throws Exception {
B b = new B();
b = null;
System.gc();
Thread.sleep(500);
}
}
class B extends A {
@Override
public void finalize() {
System.out.println("B finalize by the finalize method");
}
}
输出:
B finalize by the finalize method
A finalize by the finalizerGuardian