单例模式】单例可能被破坏的场景有哪些?

2022-05-07 18:07发布

4条回答
征戰撩四汸
2楼 · 2022-05-09 17:36

1、反射对单例模式的破坏

采用反射的方式另辟蹊径实例了该类,导致程序中会存在不止一个实例。

解决方案

其思想就是采用一个全局变量,来标记是否已经实例化过了,如果已经实例化过了,第 二次实例化的时候,抛出异常


2、clone()对单例模式的破坏

当需要实现单例的类允许clone()时,如果处理不当,也会导致程序中出现不止一个实例

解决方案

重写clone()方法,调clone()时直接返回已经实例的对象


3、序列化对单例模式的破坏

在使用序列化/反序列化时,也会出现产生新实例对象的情况

解决方案

在反序列化时的回调方法 readResolve()中返回单例对象


4、指令重排导致懒汉式单例模式失效

看似简单的一段赋值语句:instance = new Singleton();,其实JVM内部已经转换为多条指令:

memory = allocate(); //1:分配对象的内存空间

ctorInstance(memory); //2:初始化对象

instance = memory; //3:设置instance指向刚分配的内存地址

但是经过重排序后如下:

memory = allocate(); //1:分配对象的内存空间

instance = memory; //3:设置instance指向刚分配的内存地址,此时对象还没被初始化

ctorInstance(memory); //2:初始化对象

可以看到指令重排之后,instance指向分配好的内存放在了前面,而这段内存的初始化被排在了后面,在线程A初始化完成这段内存之前,线程B虽然进不去同步代码块,但是在同步代码块之前的判断就会发现instance不为空,此时线程B获得instance对象进行使用就可能发生错误.

解决方案
private static volatile Singleton instance = null;


摩羯摩羯
3楼 · 2022-05-10 09:26

多线程破坏单例 在多线程环境下,线程的时间片是由CPU自由分配的,具有随机性,而单例对象作为共享资源可能会同时被多个线程同时操作,从而导致同时创建多个对象。当然,这种情况只出现在懒汉式单例中。如果是饿汉式单例,在线程启动前就被初始化了,不存在线程再创建对象的情况

猫的想法不敢猜
4楼 · 2022-05-12 13:58

我把可能出现单例被破坏的情况,一共归纳为五种,分别为多线程破坏单例、指令重排破坏单例、克隆破坏单例、反序列化破坏单例、反射破坏单例。

来源于网络,仅供参考

回答: 2022-05-12 13:58

我把可能出现单例被破坏的情况,一共归纳为五种,分别为多线程破坏单例、指令重排破坏单例、克隆破坏单例、反序列化破坏单例、反射破坏单例。

来源于网络,仅供参考

回答: 2022-05-19 17:22

image.png

小光光321
5楼 · 2022-05-24 09:41

1.反射对单例模式的破坏 采用反射的方式另辟蹊径实例了该类,导致程序中会存在不止一个实例。 解决方案 其思想就是采用一个全局变量,来标记是否已经实例化过了,如果已经实例化过了,第 二次实例化的时候,抛出异常

2.clone()对单例模式的破坏 当需要实现单例的类允许clone()时,如果处理不当,也会导致程序中出现不止一个实例 解决方案 重写clone()方法,调clone()时直接返回已经实例的对象

3.序列化对单例模式的破坏 在使用序列化/反序列化时,也会出现产生新实例对象的情况 解决方案 在反序列化时的回调方法 readResolve()中返回单例对象


回答: 2022-05-26 10:59

多线程破坏单例 在多线程环境下,线程的时间片是由CPU自由分配的,具有随机性

改为DCL双重检查锁的写法。

使用静态内部类的写法,性能更高。 第二种:指令重排破坏单例 指令重排也可能导致懒汉式单例被破坏。来看这样一句代码: instan...

指令重排破坏单例 指令重排也可能导致懒汉式单例被破坏。来看这样一句代码: insta...

克隆破坏单例 在Java中,所有的类就继承自Object,


相关问题推荐

  • 回答 5

     优点: 一、实例控制  单例模式会阻止其他对象实例化其自己的单例对象的副本,从而确保所有对象都访问唯一实例。 二、灵活性  因为类控制了实例化过程,所以类可以灵活更改实例化过程。 缺点: 一、开销  虽然数量很少,但如果每次对象请求引用时都要...

没有解决我的问题,去提问