`
1025250620
  • 浏览: 225389 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java单例 序列化陷阱~

 
阅读更多

序列化是java中一个很常用而且很强大的功能。个人的看法,将java对象保存到磁盘,以后再从磁盘中读出来,这是java最常用到的功能之一。在基本的情况下,序列化能够“简单的起作用(just work)”。然而,随着越来越复杂的对象格式以及设计模式的被采用,透明的对象(transparent object)序列化可以“简单的起作用(just work)”的可能性变得越来越不可能了。在处理一个可控制集合的实例,比如单例和enum,就是序列化需要一些而外帮助的一种场景。 

在单例对象可序列的任何场景,确保单例的对象被使用了是非常重要的。这是通过readresolve()接口来实现的。单例就是一个很好的例子:

 

  1. public final class MySingleton {  
  2.  private MySingleton() { }  
  3.  private static final MySingleton INSTANCE = new MySingleton();  
  4.  public static MySingleton getInstance() { return INSTANCE; }  
  5. }  
 

 

在上面的例子中,仅有一种方法获得MySingleton的实例-那就是使用getInstance()这个方法。然后,在简单的添加一个接口的实现后,这段代码就变得不可用了:

 

  1. public final class MySingleton implements Serializable {  
  2. //.  
 

 

现在通过可序列化的工具,我们可以将一个单例的实例对象写到磁盘,然后再读回来,从而有效地获得一个实例。即使构造函数是私有的,可序列化工具依然可以通过特殊的途径去创建类的一个新的实例。序列化操作提供了一个很特别的钩子(hook)-类中具有一个私有的被实例化的方法readresolve(),这个方法可以确保类的开发人员在序列化将会返回怎样的object上具有发言权。足够奇怪的,readresolve()并不是静态的,但是在序列化创建实例的时候被引用。我们在一分钟内就开始体验这个。下面的例子将说明readresolve()怎样在我们的单例中起作用:

 

  1. public final class MySingleton {  
  2.  private MySingleton() { }  
  3.  private static final MySingleton INSTANCE = new MySingleton();  
  4.  public static MySingleton getInstance() { return INSTANCE; }  
  5.  private Object readResolve() throws ObjectStreamException {  
  6.   // instead of the object we're on,   
  7.   // return the class variable INSTANCE  
  8.   return INSTANCE;   
  9.  }  
  10. }  
 

 

目前为止情形还是蛮不错的。但是在处理多个实例的事情,情形就变得有点复杂了。为了解释这点,我将通过一个类型安全的enmumeration来表现这点。请记住,JDK 5的enum类型已经帮你自动处理了这个readresolve的情况。下面是一个很小的enumeration的例子:

 

  1. public final class Sides {  
  2.  private int value;  
  3.  private Sides(int newVal) { value = newVal; }  
  4.  private static final int LEFT_VALUE = 1;  
  5.  private static final int RIGHT_VALUE = 2;  
  6.  private static final int TOP_VALUE = 3;  
  7.  private static final int BOTTOM_VALUE = 4;  
  8.    
  9.  public static final LEFT = new Sides(LEFT_VALUE);  
  10.  public static final RIGHT = new Sides(RIGHT_VALUE);  
  11.  public static final TOP = new Sides(TOP_VALUE);  
  12.  public static final BOTTOM = new Sides(BOTTOM_VALUE);  
  13.    
  14. }  
 

 

现在,我们来实现序列化,用来决定将返回那个实例的key取决于对象本身被设定的值:

 

  1. public final class Sides implements Serializable {  
  2.  private int value;  
  3.  private Sides(int newVal) { value = newVal; }  
  4.  private static final int LEFT_VALUE = 1;  
  5.  private static final int RIGHT_VALUE = 2;  
  6.  private static final int TOP_VALUE = 3;  
  7.  private static final int BOTTOM_VALUE = 4;  
  8.    
  9.  public static final LEFT = new Sides(LEFT_VALUE);  
  10.  public static final RIGHT = new Sides(RIGHT_VALUE);  
  11.  public static final TOP = new Sides(TOP_VALUE);  
  12.  public static final BOTTOM = new Sides(BOTTOM_VALUE);  
  13.    
  14.  private Object readResolve() throws ObjectStreamException {  
  15.   // Switch on this instance's value to figure out which class variable  
  16.   // this is meant to match  
  17.   switch(value) {  
  18.    case LEFT_VALUE: return LEFT;  
  19.    case RIGHT_VALUE: return RIGHT;  
  20.    case TOP_VALUE: return TOP;  
  21.    case BOTTOM_VALUE: return BOTTOM;    
  22.   }  
  23.   return null;  
  24.  }  
  25. }  
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics