我之前发布了关于java.lang.Object
类及其方法的一系列文章。在介绍了Object
之后,我们又探究了clone()
和euqals()
方法。在这篇文章中,我们将继续讨论Object
中的finalize()
,getClass()
,和hashCode()
方法。
终止
问: finalize()
方法是用来做什么的?
答: finalize()
方法可以被子类对象所覆盖,然后作为一个终结者,当GC被调用的时候完成最后的清理工作(例如释放系统资源之类)。这就是终止。默认的finalize()
方法什么也不做,当被调用时直接返回。
对于任何一个对象,它的
finalize()
方法都不会被JVM执行两次。如果你想让一个对象能够被再次调用的话(例如,分配它的引用给一个静态变量),注意当这个对象已经被GC回收的时候,finalize()
方法不会被调用第二次。
问: 有人说要避免使用finalize()
方法,这是真的吗?
答: 通常来讲,你应该尽量避免使用finalize()
。相对于其他JVM实现,终结器被调用的情况较少——可能是因为终结器线程的优先级别较低的原因。如果你依靠终结器来关闭文件或者其他系统资源,可能会将资源耗尽,当程序试图打开一个新的文件或者新的系统资源的时候可能会崩溃,就因为这个缓慢的终结器。
问: 应该使用什么来替代终结器?
答: 提供一个明确的用来销毁这个对象的方法(例如,java.io.FileInputStream
的void close()
方法),并且在代码中使用try - finally
结构来调用这个方法,以确保无论有没有异常从try
中抛出,都会销毁这个对象。参考下面释放锁的代码:
1 2 3 4 5 6 7 8 9 10 |
Lock l = ...; // ... is a placeholder for the actual lock-acquisition code l.lock(); try { // access the resource protected by this lock } finally { l.unlock(); } |
这段代码保证了无论try
是正常结束还是抛出异常都会释放锁。
问: 什么情况下适合使用终结器?
答: 终结器可以作为一个安全保障,以防止声明的终结方法(像是java.io.FileOutputStream
对象的close()
方法或者java.util.concurrent.Lock
对象的Lock()
方法)没有被调用。万一这种情况出现,终结器可以在最后被调用,释放临街资源。
问: 怎么写finalize()
?
答: 可以遵循下面这个模式写finalize()
方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
@Override protected void finalize() throws Throwable { try { // Finalize the subclass state. // ... } finally { super.finalize(); } } |
子类终结器一般会通过调用父类的终结器来实现。当被调用时,先执行try
模块,然后再在对应的finally
中调用super.finalize()
;这就保证了无论try
会不会抛出异常父类都会被销毁。
问: 如果finalize()
抛出异常会怎样?
答: 当finalize()
抛出异常的时候会被忽略。而且,对象的终结将在此停止,导致对象处在一种不确定的状态。如果另一个进程试图使用这个对象的话,将产生不确定的结果。通常抛出异常将会导致线程终止并产生一个提示信息,但是从finalize()
中抛出异常就不会。
问: 我想实践一下finalize()
方法,能提供一个范例吗?
答: 参考代码清单1.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
class LargeObject { byte[] memory = new byte[1024*1024*4]; @Override protected void finalize() throws Exception { System.out.println("finalized"); } } public class FinalizeDemo { public static void main(String[] args) { while (true) new LargeObject(); } } |
代码清单1:实践finalize()
代码清单1中的代码写了一个FinalizeDemo
程序,重复地对largeObject
类实例化。每一个Largeobject
对象将产生4M的数组。在这种情况下,由于没有指向该对象的引用,所以LargeObject
对象将被GC回收。
GC会调用对象的finalize()
方法来回收对象。LargeObject
重载的finalize()
方法被调用的时候会想标准输出流打印一条信息。它没有调用父类的finalize()
方法,因为它的父类是Object
,即父类的finalize()
方法什么也不做。
编译(javac FinalizeDemo.java
)并运行(java FinalizeDemo
)代码清单1.当我在我的环境下(64位win7平台)使用JDK7u6来编译运行的时候,我看到一列finalized
的信息。但是在JDK8的环境下时,在几行finalized
之后抛出了java.lang.OutOfMemoryError
。
因为
finalize()
方法对于虚拟机来说不是轻量级的程序,所以不能保证你一定会在你的环境下观察到输出信息。
得到对象的类
问: getClass()
方法是用来做什么的?
答: 通过getClass()
方法可以得到一个和这个类有关的java.lang.Class
对象。返回的Class
对象是一个被static synchronized
方法封装的代表这个类的对象;例如,static sychronized void foo(){}
。这也是指向反射API。因为调用getClass()
的对象的类是在内存中的,保证了类型安全。
问: 还有其他方法得到Class
对象吗?
答: 获取Class
对象的方法有两种。可以使用类字面常量,它的名字和类型相同,后缀位.class;例如,Account.class
。另外一种就是调用Class
的foeName()
方法。类字面常量更加简洁,并且编译器强制类型安全;如果找不到指定的类编译就不会通过。通过forname()
可以动态地通过指定包名载入任意类型地引用。但是,不能保证类型安全,可能会导致Runtime
异常。
问: 实现equals()
方法的时候,getClass()
和instanceof
哪一个更好?
答: 使用getClass()
还是instanceof
的话题一直都是Java社区争论的热点,Angelika Langer的Secrets of equals – Part 1这片文章可以帮助你做出选择。关于正确覆盖equals()
方法(例如保证对称性)的讨论,Lang的这篇文章可以作为一个很好的参考手册。