双重检查锁实现单例需用volatile修饰实例,防止指令重排序导致线程看到未初始化对象;标准写法含两次null检查与synchronized块;推荐静态内部类或枚举替代。
用双重检查锁(Double-Checked Locking)实现线程安全的单例,核心是减少同步开销,同时保证实例只被创建一次且可见性正确。关键在于 volatile 修饰实例变量,防止指令重排序导致其他线程看到未初始化完成的对象。
在没有 volatile 时,JVM 可能将单例对象的构造过程(分配内存 → 初始化 → 赋值给静态引用)重排序为:分配内存 → 赋值给静态引用 → 初始化。此时另一个线程可能读到非 null 的引用,但对象尚未初始化完毕,造成空指针或异常状态。
加上 volatile 后,禁止了该重排序,并确保后续读操作能看到初始化后的全部字段值(happens-before 语义)。
以下是推荐的实现方式:
public class Singleton {
private static volatile Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
if (instance == null) { // 第一次检查(无锁)
synchronized (Singleton.class) {
if (instance == null) { // 第二次检查(加锁后)
instance = new Singleton(); // volatile 保证原子性和可见性
}
}
}
return instance;
}
}
注意点:

双重检查锁虽经典,但易出错。以下方式更推荐:
以下写法是错误或不安全的:
双重检查锁不是最简单的方式,但理解它有助于掌握并发编程中的可见性、有序性与同步粒度问题。实际项目中,优先考虑静态内部类或枚举方式。