Java 单例模式的安全实现方式

来源:互联网 发布:插值与数据拟合模型 编辑:程序博客网 时间:2024/05/18 02:27

1. 通过双重检查加锁机制实现线程安全的单例模式

/** *  * 双重检查加锁机制:实现线程安全的单例模式 */public class Singleton {/** * 双重检查加锁机制的实现一般会使用一个关键字volatile,它的意思是:被volatile * 修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。 *  * 提示:由于volatile关键字可能会屏蔽掉虚拟机中的一些必要的代码优化,所以运行效率并不是很高, * 因此一般建议,没有特别的需要,不要使用。也就是说,虽然可以使用“双重检查加锁”机制实现线程安全的实例, 但并不建大量采用,可以根据情况来采用 */private volatile static Singleton instance = null;private Singleton() {}public static Singleton getInstance() {// 先检查实例是否存在,如果不存在才进入下面的同步块if (instance == null) {// 同步块,线程安全地创建实例synchronized (Singleton.class) {// 再次检查实例是否存在,如果不存在才真正地创建实例instance = new Singleton();}}return instance;}}


2. 一种更好的安全单例实现方式

/** * 一种更好的安全单例实现方式:既能实现延迟加载,又能实现线程安全 *  * 这个实现方式综合使用了Java的类级内部类和多线程缺省同步锁的知识,很巧妙地同时实现了延迟加载和线程安全 */public class Singleton2 {/** * 类级的内部类,也就是静态的成员式内部类,该内部类的实例与外部类的实例没有绑定关系,而且只有被调用到时才会装载,从而实现了延迟加载 *  */private static class SingletonHolder {/** * 静态初始化器,由JVM来保证线程安全 */private static Singleton2 instance = new Singleton2();}/** * 私有化构造方法 */private Singleton2() {}/** * 这个模式的优势在于:getInstance方法并没有被同步,并且只是执行一个域的访问,因此延迟初始化并没有增加任何访问成本 *  */public static Singleton2 getInstance() {return SingletonHolder.instance;}}

0 0