避免使用终结方法(finalizer)

终结方法(finalizer)通常是不可预测的,也是很危险的,一般情况下是不必要的。

不要把finalizer当成C++中析构函数的对应物。java中,当对象不可达时(即没有引用指向这个对象时),会由垃圾回收器来回收与该对象相关联的内存资源;而其他的内存资源,则一般由try-finally代码块来完成类似的工作。

一、finalizer的缺点:

1. 终结方法的缺点在于不能保证会被及时地执行。

及时执行finalizer方法是JVM垃圾回收方法的一个主要功能。由于不同JVM的垃圾回收算法不同,JVM会“非故意的”延迟执行终结方法,因此终结方法的执行时间点是非常不稳定的。

2.finalizer方法的线程优先级比当前程序的其他线程优先级要低,且JAVA语言规范不保证哪个线程可以执行finalizer方法。

3.JAVA语言规范不仅不保证及时执行finalizer方法,还不保证一定会执行finalizer方法。当程序终止时,有可能一些对象的finalizer方法还没有执行。——不应该依赖finalizer方法来更新重要的持久状态。

4.System.gc和System.runFinalization不保证finalizer一定执行。

5.System.runFinalizersOnExit Runtime.runFinalizersOnExit 可以保证finalizer一定执行,但是这两个方法已经废弃。

6.如果未捕获的异常在finalizer方法中抛出来,这个异常可以被忽略(警告都不会打印出来),且finalizer方法会终止。这样这个异常就使对象处于“被破坏”的状态,如果另一个线程要使用这个对象,就可能发生不确定的行为。

7.finalizer方法会有非常严重的(Severe)性能损失

二、不用finalizer方法,怎么来实现线程中对象资源的终止呢?

使用显示终止方法。

显示终止方法的要求:

1.实例必须记录下自己是否已经被终止了

2.显示终止方法必须在一个私有域中记录下“该对象已经不再有效”

3.在执行终止方法之后,执行对象其他方法时要检查“该对象已经不再有效”私有域,抛出IllegalStateException。

显示终止方法的例子:

1.InputStream

2.OutputStream

3.java.sql.Connection

4.java.util.Timer

显示终止方法的使用方法:通常和try-finally一起使用,以确保及时终止。

FileInputStream fileInputStream = new FileInputStream();

try{

//Do something about fileInputStream;

}finally{

fileInputStream.close();

}

三、终结方法的合法用途。

1.作为安全网——显示终止方法忘记调用的时候

2.终止非关键的本地资源(android JNI操作中)

四、终结方法的执行过程中要保证:如果子类的终结过程出现异常,超类的终结过程也会得到执行。

由于终结方法链不会自动执行,因此我们需要手动保证这一点。

方法一:使用try – finalize 代码结构

@Override 
protected void finalize() throws Throwable { 
    try{ 
        ...//Finalize subclass state 
    } finally { 
        super.finalize(); 
    } 
}

方法二:使用finalizer guardian(终结方法守卫者)

class A {

//终结守卫者
    private final Object finalizerGuardian = new Object() {

@Override
        //终结守卫者的终结方法将被执行
        protected void finalize() {

//finalizer这outer class ——A
            System.out.println("A finalize by the finalizerGuardian");
        }
    };

总之,

1.除非是作为安全网或者是为了终结非关键的本地资源,否则请不要使用终结方法。

2.如果确实需要,可以使用显示终止方法

2.如果没办法真的使用了finalize,别忘记了调用super.finalize()。还可以考虑是否使用终结方法守卫者,使未调用super.finalize()方法的类的父类的终结方法也会被执行。

参考资料:

《Effective Java 第二版》

最新文章

  1. 控制对话框风格的activity的显示大小与位置
  2. cs
  3. QAbstractItemView::setRootIndex(const QModelIndex & index) 失效
  4. (转)高性能网站架构之缓存篇—Redis集群搭建
  5. poj1611(并查集)
  6. scala学习心得3
  7. WPF SDK研究 之 AppModel
  8. 微信JS SDK使用权限签名算法
  9. sql server中局部变量与全局变量的 申明与赋值(转)
  10. Linux内核完全注释之编程语言和环境(一)
  11. inline-block元素的一些坑
  12. DTD - Attributes
  13. JSP自定义标签——简单标签(1)
  14. Swiper.js使用遇到的问题总结onSlideChangeEnd回调偶尔触发,偶尔不触发等;
  15. Entity Framework技术系列之0:开篇
  16. 2018-2019-2 网络对抗技术 20165311 Exp2 后门原理与实践
  17. 2019-oo-第二次总结
  18. 【java多线程】队列系统之LinkedBlockingDeque源码
  19. oracle 与其他数据库如mysql的区别
  20. Day 07 文件的相关操作

热门文章

  1. [BZOJ 1801] [Ahoi2009]chess 中国象棋 【DP】
  2. hdu 5150 Sit sit sit
  3. Black Box《优先队列》
  4. java实战之数组工具集
  5. 【转】深入了解android平台的jni---注册native函数
  6. Java系统变量设置方式
  7. 用数组实现栈demo
  8. POJ 2029 Get Many Persimmon Trees(水题)
  9. Java EL 详细用法讲解
  10. A - 棋盘问题