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