singleton模式四种线程安全的实现

本文详细介绍了单例模式的概念、特点及应用,包括饿汉式、懒汉式、双检锁和持有类四种实现方式,适合对单例模式感兴趣的技术人员阅读。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.描述:

    Singleton(单例)是设计模式的一种,为了保证一个类仅有一个实例,并提供一个访问它的全局访问点。

 

2.主要特点:     

    1)单例类确保自己只有一个实例(构造函数私有:不被外部实例化,也不被继承)。

    2)单例类必须自己创建自己的实例。

    3)单例类必须为其他对象提供唯一的实例。

 

3.单例模式的应用:

    资源管理器,回收站,打印机资源,线程池,缓存,配置信息类,管理类,控制类,门面类,代理类通常被设计为单例类

    如果程序有多个类加载器又同时使用单例模式就有可能多个单例并存就要找相应解决方法了

 

4.实现方法:

如果应用程序总是创建并使用单例实例或在创建和运行时开销不太

    1).Eager initialization 饿汉式单例类(依赖jvm在加载类时创建唯一单例实例)

 

public class EagerSingleton {
        // jvm保证在任何线程访问uniqueInstance静态变量之前一定先创建了此实例
        private static EagerSingleton uniqueInstance = new EagerSingleton();

        // 私有的默认构造子,保证外界无法直接实例化
        private EagerSingleton() {
        }

        // 提供全局访问点获取唯一的实例
        public static EagerSingleton getInstance() {
                return uniqueInstance;
        }
}

  

 

 如果开销比较大,希望用到时才创建就要考虑延迟实例化,或者Singleton的初始化需要某些外部资源(比如网络或存储设备),就要用后面的方法了

    2)Lazy initialization 懒汉式单例类

public class LazySingleton {
        private static LazySingleton uniqueInstance;

        private LazySingleton() {
        }

        public static synchronized LazySingleton getInstance() {
                if (uniqueInstance == null)
                        uniqueInstance = new LazySingleton();
                return uniqueInstance;
        }
}

 

 

同步一个方法可能造成程序执行效率下降100倍,完全没有必要每次调用getInstance都加锁,事实上我们只想保证一次初始化成功,其余的快速返回而已,如果在getInstance频繁使用的地方就要考虑重新优化了

 

    3)"双检锁"(Double-Checked Lock)尽量将"加锁"推迟,只在需要时"加锁"(仅适用于java 5.0 以上版本,volatile保证原子操作) 
      happens-before:"什么什么一定在什么什么之前运行",也就是保证顺序性.
      现在的CPU有乱序执行的能力(也就是指令会乱序或并行运行,可以不按我们写代码的顺序执行内存的存取过程),并且多个CPU之间的缓存也不保证实时同步,只有上面的happens-before所规定的情况下才保证顺序性.

      JVM能够根据CPU的特性(CPU的多级缓存系统、多核处理器等)适当的重新排序机器指令,使机器指令更符合CPU的执行特点,最大限度的发挥机器的性能.

      如果没有volatile修饰符则可能出现一个线程t1的B操作和另一线程t2的C操作之间对instance的读写没有happens-before,可能会造成的现象是t1的B操作还没有完全构造成功,但t2的C已经看到instance为非空,这样t2就直接返回了未完全构造的instance的引用,t2想对instance进行操作就会出问题.

    volatile 的功能:
      1. 避免编译器将变量缓存在寄存器里 
      2. 避免编译器调整代码执行的顺序

优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份

public class DoubleCheckedLockingSingleton {
        // java中使用双重检查锁定机制,由于Java编译器和JIT的优化的原因系统无法保证我们期望的执行次序。
        // 在java5.0修改了内存模型,使用volatile声明的变量可以强制屏蔽编译器和JIT的优化工作
        private volatile static DoubleCheckedLockingSingleton uniqueInstance;

        private DoubleCheckedLockingSingleton() {
        }

        public static DoubleCheckedLockingSingleton getInstance() {
                if (uniqueInstance == null) {
                        synchronized (DoubleCheckedLockingSingleton.class) {
                                if (uniqueInstance == null) {
                                        uniqueInstance = new DoubleCheckedLockingSingleton();
                                }
                        }
                }
                return uniqueInstance;
        }
}

 

    4)Lazy initialization holder class 满足所有 Double-Checked Locking 满足的条件,并且没有显示的同步操作

public class LazyInitHolderSingleton {
        private LazyInitHolderSingleton() {
        }

        private static class SingletonHolder {
                private static final LazyInitHolderSingleton INSTANCE = new LazyInitHolderSingleton();
        }

        public static LazyInitHolderSingleton getInstance() {
                return SingletonHolder.INSTANCE;
        }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值