序列化-理解readResolve()

 

序列化-理解readResolve()

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

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

view plaincopy to clipboardprint?

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()这个方法。然后,在简单的添加一个接口的实现后,这段代码就变得不可用了:

view plaincopy to clipboardprint?

1.    public final class MySingleton implements Serializable {   

2.    //.  

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

view plaincopy to clipboardprint?

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的例子:

view plaincopy to clipboardprint?

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取决于对象本身被设定的值:

 

view plaincopy to clipboardprint?

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. }  

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值