《Java更多的库谜题83:诵读困难者的一神论.docx》由会员分享,可在线阅读,更多相关《Java更多的库谜题83:诵读困难者的一神论.docx(3页珍藏版)》请在金锄头文库上搜索。
1、 Java更多的库谜题83:诵读困难者的一神论public class Dog extends Exception public static final Dog INSTANCE = new Dog(); private Dog() public String toString() return “Woof“; 结果证明这个人的做法是错误的。你能够在这个类的外部不使用反射来创立出第2个Dog实例吗? 这个类可能看起来像一个单件,但它并不是。问题在于,Dog扩展了Exception,而Exception实现了java.io.Serializable。这就意味着Dog是可序列化的(seriali
2、zable),并且解序列(deserialization)会创立一个隐蔽的构造器。正如下面的这段程序所演示的,假如你序列化了Dog.INSTANCE,然后对得到的字节序列(byte sequence)进展解序列,最终你就会得到另外一个Dog。该程序打印的是false,表示新的Dog实例和原来的那个实例是不同的,并且它还打印了Woof,说明新的Dog实例也具有相应的功能:import java.io.*;public class CopyDog / Not to be confused with copycat public static void main(String args) Dog n
3、ewDog = (Dog) deepCopy(Dog.INSTANCE); System.out.println(newDog = Dog.INSTANCE); System.out.println(newDog); / This method is very slow and generally a bad idea! static public Object deepCopy(Object obj) try ByteArrayOutputStream bos = new ByteArrayOutputStream(); new ObjectOutputStream(bos).writeOb
4、ject(obj); ByteArrayInputStream bin = new ByteArrayInputStream(bos.toByteArray(); return new ObjectInputStream(bin).readObject(); catch(Exception e) throw new IllegalArgumentException(e); 要订正这个问题,可在Dog中添加一个readResolve方法,它可以将那个隐蔽的构造器转变为一个隐蔽的静态工厂(static factory),以返回原来那个的Dog EJ Items 2,57。在Dog中添加了这个方法之
5、后,CopyDog将打印true而不是false,表示那个“复本”实际上就是原来的那个实例: private Object readResolve() / Accept no substitues! return INSTANCE; 这个谜题的主要教训就是一个实现了Serializable的单件类,必需有一个readResolve方法,用以返回它的的实例。一个次要的教训就是,有可能由于对一个实现了Serializable的类进展了扩展,或者由于实现了一个扩展自Serializable的接口,使得我们在无意中实现了Serializable。给平台设计者的教训是,隐蔽的构造器,例如序列化中产生的那个,会让读者对程序行为的产生错觉。