单例模式(Singleton Pattern)是一种设计模式,它的目的是确保一个类在整个程序的生命周期内只有一个实例,并且提供全局访问点来获取该实例。
实现方式
懒加载 - 懒汉模式
public class Singleton {
private volatile static Singleton instance;
private Singleton() {
}
public static Singleton getInstance() {
if (instance == null) { // 第一次检查
synchronized (Singleton.class) {
if (instance == null) { // 第二次检查
instance = new Singleton();
}
}
}
return instance;
}
}
这里面用了双重检查锁实现了线程安全的单例模式 ,第一次检查看 instance 是否为空,如果为空,就进行第二次检查。在两次检查之间我对 Singletin.class 加锁,这是个类对象,在类加载的时候会生成,是唯一的。
所以如果有多个线程都通过了第一次检查,那么在这就保证只有一个线程可以进入同步块。进入之后再进行一次检查,因为这个时候只有一个线程在同步块,所以检查判定成功的话,就创建实例,而且只会创建一个,不会有并发问题。
使用双重检查所可以保证在线程安全前提下去创建对象,保证单例。
使用synchronized
关键字会有一定的性能开销,但是通过双重检查锁定,只在必要时才进行同步,减少了性能损失。
预加载 - 饿汉模式
public class Singleton {
private static final Singleton INSTANCE = new Singleton();
private Singleton() {}
public static Singleton getInstance() {
return INSTANCE;
}
}
预加载模式是在类加载的时候就创建了单例实例。在预加载模式下,Singleton
类被加载时,静态成员 INSTANCE
会被立即初始化,实例会在类加载的同时创建。
静态内部类模式
在这种实现中,静态内部类的实例是在运行时第一次被访问时创建的,而不是在编译时或类被加载时。
public class Singleton {
private Singleton() {}
private static class SingletonHolder {
private static final Singleton INSTANCE = new Singleton();
}
public static Singleton getInstance() {
return SingletonHolder.INSTANCE;
}
}
Singleton
类被加载时,静态内部类 SingletonHolder
不会被立即加载和初始化。
只有当 getInstance()
方法第一次被调用时,SingletonHolder
类才会被加载和初始化,此时会创建 INSTANCE
对象。
枚举模式
public enum Singleton {
INSTANCE;
public void doSomething() {
// 业务逻辑
}
}
枚举模式是实现单例的最佳方法
枚举模式被认为是实现单例的最佳方法,因为它具备以下特点:
- 线程安全:枚举实例在类加载时由JVM自动初始化,天然线程安全,无需额外同步处理。
- 防止反射攻击:枚举的构造器是私有的,Java反射机制无法调用,避免了反射破坏单例。
- 防止序列化破坏:枚举类型在序列化和反序列化过程中,由JVM保证实例的唯一性,不会生成新的实例。
应用场景
饿汉式:适合资源消耗低且实例需要频繁使用的场景。因为无论实例用不用都会被先加载出来。
双重检查加锁--懒汉式:适合需要延迟加载的场景。
静态内部类:适合需要延迟加载且追求简单高效的多线程环境。
枚举式:适合要求高安全性(防反射、防序列化破坏)且实现简洁的场景。