使用场景
多线程
目的
为了提高效率,因为我们知道 synchronized 关键字不可避免的影响了程序执行的效率,而改进的思路之一就是让 synchronized 的作用范围变小,双重监测就是基于这样的一个思路
代码示例
单例模式在多线程情况下的最近解决方案
class Signal{
private Signal() {
}
private volatile static Signal instance;
private static Signal getInstance(){
if (instance == null) {
synchronized (Signal.class){
if (instance == null) {
instance = new Signal();
}
}
}
return instance;
}
}
理解
如果instance检测到不为null,那么它一定被实例化了,volatile修饰的变量,在值被修改后,会立即同步,其他线程读到的就是修改后的值,但是instance被检测到null,它不一定被实例化,可能两个线程同时检测到null,同时准备实例化,所以,在此基础上,加了一把锁
简而言之:双重检测外层的检查是粗检测,排除了逻辑的不可能,而内层的检测是细检测,保证了绝对了安全