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