双重检查锁通过两次判空和volatile保证线程安全与性能,避免多线程下重复创建实例,同时实现懒加载。

在多线程环境下,单例模式的实现必须确保只有一个实例被创建,且多个线程访问时不会出现竞态条件。Java中常用“双重检查锁”结合“懒加载”来实现线程安全的单例模式,既能保证性能,又能延迟初始化。
为什么需要双重检查锁?
如果只使用synchronized修饰整个获取实例的方法,虽然线程安全,但每次调用都会加锁,影响性能。而单纯的懒加载(方法内判断实例为空再创建)在多线程下可能产生多个实例。
双重检查锁通过两次判断实例是否为空,减少不必要的同步开销:
- 第一次检查:避免已创建实例后仍进入同步块
- 第二次检查:防止多个线程同时通过第一层检查,导致重复创建
如何正确实现双重检查锁单例?
关键点包括:volatile关键字、私有构造函数、静态实例变量和同步控制。以下是标准写法:
立即学习“Java免费学习笔记(深入)”;
public class Singleton {
// 使用volatile确保多线程下instance的可见性和禁止指令重排序
private static volatile Singleton instance;
// 私有构造函数防止外部实例化
private Singleton() {}
// 获取唯一实例的方法
public static Singleton getInstance() {
if (instance == null) { // 第一次检查
synchronized (Singleton.class) { // 加锁
if (instance == null) { // 第二次检查
instance = new Singleton(); // 创建实例
}
}
}
return instance;
}
}
注意:volatile在这里至关重要。它防止JVM在对象未完全初始化前将引用赋值给instance,从而避免其他线程拿到一个不完整的对象。
懒加载的优势与适用场景
懒加载意味着实例在第一次被使用时才创建,节省内存资源,适合重量级对象或启动时不立即需要的组件。
- 配置管理器、日志工厂、数据库连接池等常用此模式
- 启动速度快,资源占用少
- 配合双重检查锁,兼顾线程安全与性能
替代方案简要说明
除了双重检查锁,还有几种线程安全的单例实现方式:
- 静态内部类:利用类加载机制保证线程安全,推荐使用
- 枚举单例:最简洁且防反射攻击的方式
- 饿汉式:类加载时就创建,简单但不支持懒加载
基本上就这些。双重检查锁是理解Java并发编程的重要案例,掌握其原理有助于写出高效又安全的代码。关键是不要漏掉volatile,否则在高并发下可能出问题。










