序列化-理解readResolve()

    技术2022-05-14  21

    转自:http://blog.csdn.net/haydenwang8287/archive/2010/10/25/5964130.aspx

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

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

     

    public final class MySingleton {   private MySingleton() { }   private static final MySingleton INSTANCE = new MySingleton();   public static MySingleton getInstance() { return INSTANCE; }  }    

     

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

     

    public final class MySingleton implements Serializable {  //.    

     

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

     

    public final class MySingleton {   private MySingleton() { }   private static final MySingleton INSTANCE = new MySingleton();   public static MySingleton getInstance() { return INSTANCE; }   private Object readResolve() throws ObjectStreamException {    // instead of the object we're on,     // return the class variable INSTANCE    return INSTANCE;    }  }    

     

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

     

    public final class Sides {   private int value;   private Sides(int newVal) { value = newVal; }   private static final int LEFT_VALUE = 1;   private static final int RIGHT_VALUE = 2;   private static final int TOP_VALUE = 3;   private static final int BOTTOM_VALUE = 4;      public static final LEFT = new Sides(LEFT_VALUE);   public static final RIGHT = new Sides(RIGHT_VALUE);   public static final TOP = new Sides(TOP_VALUE);   public static final BOTTOM = new Sides(BOTTOM_VALUE);     }    

     

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

     

    public final class Sides implements Serializable {   private int value;   private Sides(int newVal) { value = newVal; }   private static final int LEFT_VALUE = 1;   private static final int RIGHT_VALUE = 2;   private static final int TOP_VALUE = 3;   private static final int BOTTOM_VALUE = 4;      public static final LEFT = new Sides(LEFT_VALUE);   public static final RIGHT = new Sides(RIGHT_VALUE);   public static final TOP = new Sides(TOP_VALUE);   public static final BOTTOM = new Sides(BOTTOM_VALUE);      private Object readResolve() throws ObjectStreamException {    // Switch on this instance's value to figure out which class variable    // this is meant to match    switch(value) {     case LEFT_VALUE: return LEFT;     case RIGHT_VALUE: return RIGHT;     case TOP_VALUE: return TOP;     case BOTTOM_VALUE: return BOTTOM;      }    return null;   }  }  

     


    最新回复(0)