设计模式--单例模式(二)双重校验锁模式

来源:互联网 发布:linux批量重命名 编辑:程序博客网 时间:2024/06/06 00:14

双重检验锁模式

双重检验锁模式(double checked locking pattern),是一种使用同步块加锁的方法。程序员称其为双重检查锁,因为会有两次检查

instance == null,一次是在同步块外,一次是在同步块内。为什么在同步块内还要再检验一次?因为可能会有多个线程一起进入同步

块外的 if,如果在同步块内不进行二次检验的话就会生成多个实例了。

代码实现:

package com.jn.pro;/* * 双重校验锁 */public class SingletonClass2 {private static volatile SingletonClass2  instance;//声明成 volatileprivate SingletonClass2(){}public static SingletonClass2 getInstance(){if(instance == null){synchronized (SingletonClass2.class) {if(instance == null){instance = new SingletonClass2();}}}return instance;}}

测试类:

package com.jn.pro;/* * 双重校验锁测试类 */public class SingletonClassTest2 {public static void instantiation(){SingletonClass2 sc1 = SingletonClass2.getInstance();System.out.println("第一次取得实例sc1");SingletonClass2 sc2 = SingletonClass2.getInstance();System.out.println("第二次取得实例sc2");if(sc1 == sc2){System.out.println("sc1和sc2是同一个实例(双重校验锁模式)");}}public static void main(String[] args){instantiation();}}

运行结果:

第一次取得实例sc1第二次取得实例sc2sc1和sc2是同一个实例(双重校验锁模式)

注意:

instance = new Singleton()这句,这并非是一个原子操作,事实上在 JVM 中这句话大概做了下面 3 件事情。

    1.给 instance 分配内存
    2.调用 Singleton 的构造函数来初始化成员变量
    3.将instance对象指向分配的内存空间(执行完这步 instance 就为非 null 了)


volatile

volatile 可以禁止指令重排序优化。也就是说,在 volatile 变量的赋值操作后面会有一个内存屏障(生成的汇编代码上),读操作不会被重排序到内存屏障之前。比如上面的例子,取操作必须在执行完 1-2-3 之后或者 1-3-2 之后,不存在执行到 1-3 然后取到值的情况。

阅读全文
0 0
原创粉丝点击