单例模式-序列化破坏单例模式及解决办法

序列化破环单例模式,实例代码如下:

(1) 饿汉式单例模式,实现了序列化接口

public class SeriableSingleton implements Serializable{
    private static final SeriableSingleton instance=new SeriableSingleton();
    private SeriableSingleton(){}
    public static SeriableSingleton getInstance(){
        return instance;
    }
}

(2)  对单例模式进行序列化和反序列化破坏

public class SeriableSingletonTest {
    public static void main(String[] args) {
        SeriableSingleton s1=null;
        SeriableSingleton s2=SeriableSingleton.getInstance();
        FileOutputStream fos=null;
        ObjectOutputStream oos=null;
        FileInputStream fis=null;
        ObjectInputStream ois=null;

        try {
            fos=new FileOutputStream("SeriableSingleton.obj");
            oos=new ObjectOutputStream(fos);
            oos.writeObject(s2);
            oos.flush();
            oos.close();

            fis=new FileInputStream("SeriableSingleton.obj");
            ois=new ObjectInputStream(fis);
            s1=(SeriableSingleton)ois.readObject();
            ois.close();

            System.out.println(s1);
            System.out.println(s2);
            System.out.println(s1==s2);
        } catch (Exception e) {
            e.printStackTrace();
        }

    }
}

此段代码执行结果如下:

com.wxz.designpattern.singleton.SeriableSingleton@7ba4f24f
com.wxz.designpattern.singleton.SeriableSingleton@6d6f6e28
false

由代码执行结果可看出,对单例模式执行完序列化和反序列化后,创建了新的对象,破坏了单例模式,违背了单例模式设计初衷。

(3) 如何保证序列化方式下也能实现单例?

解决方法很简单,只需要在单例类中增加readResolve()方法,代码如下:

public class SeriableSingleton implements Serializable{
    private static final SeriableSingleton instance=new SeriableSingleton();
    private SeriableSingleton(){}
    public static SeriableSingleton getInstance(){
        return instance;
    }
    public Object readResolve(){
        return instance;
    }
}

再次执行SeriableSingletonTest测试方法,执行结果如下:

com.wxz.designpattern.singleton.SeriableSingleton@6d6f6e28
com.wxz.designpattern.singleton.SeriableSingleton@6d6f6e28
true

由此可见,增加readResolve()方法后解决了序列化方法对单例模式的破坏。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
序列和反序列可以破坏单例设计模式的安全性。当一个单例类被序列后,然后再进行反序列,会创建出一个新的实例,从而破坏了单例的特性。这是因为序列和反序列过程中会创建一个新的对象,并不会调用类的构造函数来初始新对象。因此,即使单例类被序列和反序列,也不能保证只有一个实例存在。 为了解决这个问题,可以在单例类中添加一个readResolve方法,并在该方法中返回单例实例。这样,在反序列时,就可以通过readResolve方法返回已存在的单例实例,而不是创建一个新的实例。通过这种方式,可以确保单例模式的安全性,避免了序列和反序列破坏单例的问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [深入浅出单例模式与反射与序列对单例的破坏](https://blog.csdn.net/weixin_43975523/article/details/103140654)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [设计模式|序列、反序列对单例的破坏、原因分析、解决方案及解析](https://blog.csdn.net/leo187/article/details/104332138)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值