《TIJ4》初始化与清理 练习 E10-E12 练习使用finalize()

练习10:编写一个具有finalize()方法的类,并在方法中打印消息,在main()方法中为该类创建一个对象。

package initialization;

public class E10_FinalizeCall {

    protected void finalize(){
        System.out.println("finalize() called");
    }

    public static void main(String[] args) {
        new E10_FinalizeCall();
}

注意这里并没有声明对象的引用,只有这样,垃圾回收时才会回收这个对象。
该程序中的终结器不一定会被调用,因为程序通常并没有产生足够的垃圾供回收器回收。

练习11:修改上面的程序,使得finalize()方法总会被调用。

package initialization;

public class E10_FinalizeCall {

    protected void finalize(){
        System.out.println("finalize() called");
    }

    public static void main(String[] args) {
        new E10_FinalizeCall();
        System.gc();
        System.runFinalization();   
    }
}

这里增加了两行代码:

System.gc();
System.runFinalization();

依次调用这两个方法会建议系统垃圾回收从而调用finalize(),但这不一定会发生。 调用这两个方法只会请求垃圾回收,它不能终结器一定会执行。
因此,无法保证finalize()方法总被调用。

练习12:编写Tank类,此类的状态可以是“满的”或“空的”, 其终结条件是:对象被清理时,必须处于空状态。编写finalize()以校验终结条件是否成立。在main()中测试Tank的几种使用方式。

package initialization;

public class Tank {

    static long counter;
    long id = counter++;
    boolean full;

    public Tank() {
        System.out.println(this + " created");
        full = true;
    }

    public void empty(){
        full = false;
    }

    protected void finalize(){
        if(full)
            System.out.println("Error: " + this + " must be empty at cleanup");
        else
            System.out.println(this + " cleaned up OK");
    }

    @Override
    public String toString() {
        return "Tank " + id;
    }

    public static void main(String[] args) {
        // 正确的使用和清理
        new Tank().empty();
        // 状态不为空时的清理
        new Tank();
        // 请求垃圾回收
        System.gc();
        System.runFinalization();
    }
}

这里我们创建了两个没有引用的Tank实例,因为如果我们创建了引用,那么在调用System.gc()时这些引用仍在作用域中,因此它们就不会被清理,终结器finalize()也就不会被调用。另一种方法是将引用置为null,这样也可以使引用原来指向的对象被回收。例如:

Tank tank = new Tank();
tank = null;

你永远也无法保证终结器一定会被调用,因此它们的功能就很受限。这里终结器可以用来在垃圾回收时,检测对象的状态,以确保对象可以正确得被清理。

posted @ 2018-04-11 18:03  庄浩  阅读(196)  评论(0编辑  收藏  举报