避免使用终结方法(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 第二版》

Effective Java 学习笔记之第七条——避免使用终结(finalizer)方法的更多相关文章

  1. 20145230《java学习笔记》第七周学习总结

    20145230 <Java程序设计>第7周学习总结 教材学习内容 Lambda语法概览 我们在许多地方都会有按字符串长度排序的需求,如果在同一个方法内,我们可以使用一个byName局部变 ...

  2. Effective Java 学习笔记之创建和销毁对象

    一.考虑用静态工厂方法代替构造器 1.此处的静态工厂方法是指返回指为类的对象的静态方法,而不是设计模式中的静态工厂方法. 2.静态工厂方法的优势有: a.使用不同的方法名称可显著地表明两个静态工厂方法 ...

  3. Effective Java 学习笔记----第7章 通用程序设计

    第7章 通用程序设计 第29条 将局部变量的作用域最小化     使一个局部变量的作用域最小化,最有力的技术室在第一次使用它的地方声明.   第30条 了解和使用库      效率提高.如果你不知道库 ...

  4. 201521123103 《Java学习笔记》 第七周学习总结

    一.本周学习总结 1.以你喜欢的方式(思维导图或其他)归纳总结集合相关内容. 二.书面作业 1.ArrayList代码分析 1.1 解释ArrayList的contains源代码 源代码如下: pub ...

  5. Effective Java学习笔记

    创建和销毁对象 第一条:考虑用静态工厂方法替代构造器 For example: public static Boolean valueOf(boolean b){ return b ? Boolean ...

  6. Effective Java 学习笔记之所有对象都通用的方法

    一.覆盖equals时请遵守通用约定 1.满足下列任何一个条件时,不需要覆盖equals方法 a.类的每个实例本质上都是唯一的.此时就是Object中equals方法所表达的含义. b.不关心类是否提 ...

  7. 疯狂java学习笔记之面向对象(七) - super关键字

    super有以下两大作用: 1.起限定作用:强制去访问父类的成员(Field.方法) 2.起调用作用:指定/显示调用父类的某个构造器 super调用规则: 1.子类构造器总会调用父类构造器一次,默认情 ...

  8. effective java学习笔记之不可实例化的类

    在没有显式声明一个类的构造方法时,编译器会生成默认的无参构造方法,在设计工具类时,我们通常将方法设置成静态方法,以类名.方法名的形式调用,此时这个类就没有必要创建实例,我们知道抽象类不可以被实例化,但 ...

  9. Effective Java学习笔记--创建和销毁对象

    创建和销毁对象 一.静态工厂方法代替构造器 静态工厂方法的优缺点 优点: 1.可以自定义名称(可以将功能表述的更加清晰) 2.不必每次调用都创建新的对象(同一对象重复使用) 3.返回的类型可以是原返回 ...

随机推荐

  1. bit、sbin、sfr、sfr16 区别分析

    1.bit 和 sbit 都是 C51 扩展的变量类型. bit 和 int char 之类的差不多,只不过 char=8 位, bit=1 位而已.都是变量,编译器在编译过程中分配地址.除非你指定, ...

  2. C语言宏定义使用技巧

    写好C语言,漂亮的宏定义很重要,使用宏定义可以防止出错,提高可移植性,可读性,方便性 等等.下面列举一些成熟软件中常用得宏定义...... 1.防止一个头文件被重复包含 #ifndef COMDEF_ ...

  3. ISO-7816-1-2-3协议

    第一部分:卡的电气特性一. 卡的触点分配IC卡触点的分配遵循ISO7816-2的规定,如下所示: C1 电源电压(Vcc) C5 地(GND) C2 复位信号(RST) C6 不使用 C3 时钟信号( ...

  4. Mysql分页查询

    取前5条数据 select * from table_name limit 0,5 或 select * from table_name limit 5 取第11条到第15条数据,共5条 select ...

  5. bzoj2561

    对于新加入的边,必须要既可能在最小生成树上也可能在最大生成树上我们先对于最小生成树考虑根据kruskal的理论,不难发现,u--v 长度为L的边可能出现在最小生成树上就是说删边剩下的比L小的边一定不能 ...

  6. document.getElementById的简便方式

    封装自己的元素获取方法,使元素获取变得简便 注意:1.应该要防止定义的被重写,可将同名的重新定义   2.可将封装的对象置为全局对象,方便使用 通过id查找单个元素 封装方式: //通过id查找单个元 ...

  7. QTP中FSO的使用

    序 FSO即文件系统对象(File System Object),在测试工作中有广泛的应有,它可以帮助我们自动生成测试目录,写日志,测试报告等.FSO有对象有很多属性和方法,今天只介绍几个常用的. 创 ...

  8. Sqrt(x)——LeetCode

    Implement int sqrt(int x). Compute and return the square root of x. 题目大意:实现求一个int的根. 解题思路:二分. public ...

  9. UNITY3D ShadeSH9

    UNITY3D ShadeSH9 属于Irradiance environment maps 方法,可以参考DX SDK PRTDemo,里面是几乎相同的实现,总之就是解光传输的积分方程 目前主流辐射 ...

  10. MyEclipse里项目部署到tomcat上之后,tomcat webpps文件夹里为什么找不到这个项目

         今天在MyEclipse中部署了一个java web项目,然后发现报404错误,跑到tomcat目录下的webapps文件夹里并发现没有这个项目,才发现MyEclipse没有写入webapp ...