zoukankan      html  css  js  c++  java
  • 设计模式--单例模式(二)双重校验锁模式

    双重检验锁模式

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

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

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

    代码实现:

    package com.jn.pro;
    
    /*
     * 双重校验锁
     */
    public class SingletonClass2 {
    	
    	private static volatile SingletonClass2  instance;//声明成 volatile
    	
    	private 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
    第二次取得实例sc2
    sc1和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 然后取到值的情况。

  • 相关阅读:
    一次有教益的程序崩溃调试 (中)
    读书:手工测试与自动测试
    迭代还是交付?
    用Windbg调试.NET程序的资源泄漏
    一次有教益的程序崩溃调试 (上)
    基于云计算的软件测试服务
    Vcastr 3.0 api
    学习Linux三(Linux常用命令及技巧)
    学习Linux六(Linux必学60个命令之【文件处理】)
    学习Linux四(Linux必学60个命令)
  • 原文地址:https://www.cnblogs.com/mlan/p/11060370.html
Copyright © 2011-2022 走看看