静态内部类方式是Java中实现线程安全懒加载单例的最佳选择,利用JVM类加载机制保证实例唯一性和初始化线程安全,同时实现延迟加载,代码简洁且无锁开销,优于双重检查锁定和枚举方式。

在Java中实现线程安全的懒加载单例模式,关键在于确保实例只被创建一次,并且在多线程环境下不会出现竞态条件。懒加载意味着实例在第一次被使用时才初始化,节省资源,但需要处理并发访问的问题。
使用双重检查锁定(Double-Checked Locking)
这是最常见且高效的线程安全懒加载实现方式。通过在同步代码块前后两次检查实例是否已创建,减少不必要的锁竞争。
注意:必须将实例变量声明为 volatile,防止指令重排序导致其他线程获取到未完全初始化的对象。
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();
}
}
}
return instance;
}
}
说明:
立即学习“Java免费学习笔记(深入)”;
- 第一次判空避免每次调用都进入同步块。
- synchronized 保证同一时刻只有一个线程能创建实例。
- 第二次判空防止多个线程在锁释放后重复创建。
- volatile 确保 instance 的写操作对所有线程立即可见,并禁止JVM对对象初始化过程进行重排序。
使用静态内部类实现(推荐)
利用类加载机制保证线程安全,同时实现懒加载。这种方式更简洁、安全,无需显式加锁。
public class Singleton {
private Singleton() {}
private static class Holder {
static final Singleton INSTANCE = new Singleton();
}
public static Singleton getInstance() {
return Holder.INSTANCE;
}
}
原理:
- 内部类 Holder 在外部类加载时不被初始化。
- 只有当调用 getInstance() 时才会触发 Holder 类的加载和静态变量初始化。
- 类加载过程由 JVM 保证线程安全,因此 INSTANCE 的创建天然线程安全。
枚举方式(最安全,但不支持懒加载)
Effective Java 推荐使用枚举实现单例,防止反射攻击和序列化破坏,但其初始化时机取决于类加载,无法严格控制为“首次使用”才加载。
public enum Singleton {
INSTANCE;
public void doSomething() {
// 业务方法
}
}
虽然简洁安全,但不符合“懒加载”的严格定义,因为枚举实例会在类加载阶段初始化。
对比与选择建议
- 若追求性能与真正的懒加载,推荐静态内部类方式,代码简单且无锁开销。
- 若需兼容复杂初始化逻辑且必须延迟加载,可使用双重检查锁定,但务必加上 volatile。
- 枚举适合不需要严格懒加载、注重防破坏的场景。
基本上就这些。静态内部类方式在大多数情况下是最佳选择,兼顾了线程安全、懒加载和代码简洁性。










