设计模式(一)单例模式DCL 双重校验锁

实现代码如下:

public class Singleton {

	private volatile static Singleton singleton;
	
	private Singleton() {
		
	}
	
	public static Singleton getInstance() {
		if(singleton==null) {
			synchronized(Singleton.class) {
				if(singleton==null) {
					singleton = new Singleton();
				}
			}
		}
		return singleton;
	}
}
  1. 构造方法用private修饰:防止在外部实例化多个对象。
  2. volatile作用:防止指令重排序。
    一个对象初始化的顺序大致是,在内存中开辟一块空间,初始化对象,引用指向对象地址。由于优化,可能导致指令重排序,即对象还没完全初始化完毕,引用就指向了对象地址。
    假设如下场景:
    首先singleton没有被初始化,一个线程调用getInstance()方法,判断singleton为空,进入synchronized代码块,仍为空,则开始创建singleton对象实例。对象还没初始化完,singleton这个引用就指向了没初始化完的内存地址。
    此时另一个线程也调用了getInstance()方法,这时判断singleton,已经不为空了,它指向了一个没初始化完的对象,因此直接返回了未初始化完的对象。会导致错误。
    因此用volatile修饰singleton,防止指令重排序。
  3. 第一次if(singleton==null)的作用:提升性能。
    为保证线程安全,避免多个线程创建多个singleton对象,该模式使用synchronized锁定了创建singleton对象实例的代码块。但如果singleton对象已经创建完,即不为空,则可以直接返回该对象,此时不需要锁定,提升了处理速度。
  4. 第二次if(singleton==null)的作用:避免多线程时创建多个singleton对象实例。
    假设如下场景:
    singleton对象没有被创建。此时线程A判断singleton为空,进入synchronized锁定的代码块,singleton仍为空,因此开始创建singleton对象实例。由于volatile修饰,此时singleton引用仍指向空(对象还没初始化完)。
    此时,线程B调用getInstance()方法,经过判断singleton为空,由于线程A在
    synchronized代码块中,因此线程B排队等待,线程A执行完毕后,线程B执行synchronized内的代码块,若没有第二次判断,线程B会再次new一个singleton对象。因此需要第二次判断。
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值