DCL 实现中:
1、为什么需要使用两个 if 语句?
2、为什么使用了 synchronized 关键字还需要使用 volatile 关键字?
3、双重校验锁使用需要注意的问题
public class SingletonDemo {
private static Object object = null;
}
public class SingletonDemo {
private static Object object = null;
// 初始化和获取实例
public Object getObject() {
synchronized (SingletonDemo.class) {
if (object == null) {
object = new Object();
}
return object;
}
}
}
public class SingletonDemo {
private volatile static Object object = null;
public Object getObject() {
// 如果实例已经初始化完成,直接返回实例不获取锁
if (object != null){
return object;
}
synchronized (SingletonDemo.class) {
if (object == null) {
object = new Object();
}
return object;
}
}
}
创建一个对象分为初始化和实例化两部分,大致可以分为以下几步:
1、在堆中申请一份内存
2、创建对象
3、将 object 指向我们对象的内存引用
如果没有指令重排的情况下,我们拿到的对象一定是完整的对象,但是可能存在指令重排优化,上面的顺序可能变成下面这样:
1、申请一份内存
2、将 object 指向我们对象的内存引用
3、创建对象
那么我们将会拿到一个没有实例化完成的对象,因此我们需要禁止指令重排,Java 提供了 volatile 指令来禁止指令重排。
public class SingletonDemo {
private volatile static Object object = null;
public Object getObject() {
// 如果实例已经初始化完成,直接返回实例不获取锁
if (object != null){
return object;
}
synchronized (SingletonDemo.class) {
if (object == null) {
object = new Object();
}
return object;
}
}
}
阅读量:2031
点赞量:0
收藏量:0