单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供该实例的全局访问点。
new
创建实例。getInstance()
/**
* 饿汉式单例
*/
public class EagerSingleton {
// 创建一个静态的实例,静态常量,在类加载的时候创建实例
private static final EagerSingleton instance = new EagerSingleton();
// 私有化构造方法,使的用者无法通过new关键字创建对象
private EagerSingleton() {
}
// 提供获取实例的方法
public static EagerSingleton getInstance() {
return instance;
}
}
public class demo {
public static void main(String[] args) {
EagerSingleton instance = EagerSingleton.getInstance();
EagerSingleton instance2 = EagerSingleton.getInstance();
System.out.println(instance == instance2); // true
}
}
getInstance()
时创建)。volatile
关键字(避免JVM指令重排序导致未初始化完全的错误实例)。/**
* 懒汉式单例普通实现
*/
public class LazySingleton {
private static LazySingleton instance;
private LazySingleton() {
}
/**
* 在多线程环境下,多个线程可能同时进入 if (instance == null) 条件块,
* 并且每个线程都可能执行 new LazySingleton() 语句,从而导致创建多个实例
* @return
*/
public static LazySingleton getInstance() {
if (instance == null) {
instance = new LazySingleton(); // 这里可能会有并发问题
}
return instance;
}
}
/**
* 懒汉式单例双重检查锁定实现
*/
public class LazySingleton {
// volatile 关键字是为了避免指令重排
private static volatile LazySingleton instance;
private LazySingleton() {
}
/**
* 在多线程环境下,多个线程可能同时进入 if (instance == null) 条件块,
* 并且每个线程都可能执行 new LazySingleton() 语句,从而导致创建多个实例
* 所以需要使用 synchronized 关键字修饰 getInstance() 方法,
* @return
*/
public static LazySingleton getInstance() {
if (instance == null) {
synchronized(LazySingleton.class){
if (instance == null) {
instance = new LazySingleton();
}
}
}
return instance;
}
}
/**
* 静态内部类实现单例模式
*/
public class HolderSingleton {
private HolderSingleton() {
}
/**
* JVM 确保一个类只会被初始化一次,即使多个线程同时尝试加载该类。
* 当 HolderSingleton 类被加载时,其内部的 Holder 类并不会立即被加载
*/
private static class Holder {
private static final HolderSingleton INSTANCE = new HolderSingleton();
}
/**
* 延迟加载
* 只有当调用 getInstance() 方法时,才会触发 Holder.INSTANCE 的访问
* 此时 JVM 才会加载并初始化 Holder 类,并创建 HolderSingleton 实例
* @return
*/
public static HolderSingleton getInstance() {
return Holder.INSTANCE;
}
}
Singleton.INSTANCE.doSomething()
/**
* 枚举单例模式
*/
public enum EnumSingleton {
//
/**
* 枚举实例由JVM在类加载时静态初始化,保证线程安全
* Java规范禁止通过反射创建枚举实例,避免传统单例被反射攻击的问题
* 枚举的序列化机制仅保存枚举名称,反序列化时通过valueOf还原原实例,防止生成新对象
*/
INSTANCE; // 唯一实例
/**
*实例方法
*/
public void whateverMethod() {
System.out.println("whateverMethod");
}
}
public static void main(String[] args) {
//调用
EnumSingleton.INSTANCE.whateverMethod();
}
volatile
+同步块确保安全(如支付网关)36场景 | 推荐模式 | 原因 |
---|---|---|
全局配置管理器 | 饿汉式 | 配置轻量且启动必用 |
数据库连接池 | 静态内部类 | 延迟加载避免启动卡顿,无锁线程安全 |
金融交易核心 | 枚举 | 防反射攻击,强一致性要求 |
高频工具类 | 饿汉式 | 无状态对象,快速访问 |
第三方服务代理 | 双重检查锁(DCL) | 按需加载+高并发安全(如支付网关) |
如果本文对你有帮助,点击右上角【订阅专栏】或左上角关注我
完整的23中设计模式干货,第一时间推送给你!