当前位置: 首页 > news >正文

jdk8之后都有什么优化单例的方式

一、基于 ​Lambda 表达式 + Supplier​ 的延迟初始化

利用 Supplier 函数式接口和 ​静态内部类​ 简化延迟加载逻辑:

public class LambdaSingleton {private static final Supplier<LambdaSingleton> INSTANCE = () -> {// 延迟初始化逻辑(线程安全)return Holder.INSTANCE;};private static class Holder {static final LambdaSingleton INSTANCE = new LambdaSingleton();}private LambdaSingleton() {}public static LambdaSingleton getInstance() {return INSTANCE.get();}
}

二、接口静态方法实现饿汉式单例

JDK 8 允许接口定义 ​静态方法,可用于简化饿汉式单例:

public interface ConfigService {ConfigService INSTANCE = new ConfigServiceImpl();static ConfigService getInstance() {return INSTANCE;}// 默认方法(JDK8+)default void loadConfig() {// 配置加载逻辑}
}class ConfigServiceImpl implements ConfigService {// 具体实现
}

三、模块化(JDK9+)​​ 增强单例安全性

通过 Java 模块系统(module-info.java)限制反射访问,防止恶意代码破坏单例:

module com.example.singleton {exports com.example.singleton;  // 仅暴露公共APIopens com.example.singleton.impl to spring.core; // 限制反射访问范围
}

四、VarHandle(JDK9+)​​ 优化双重校验锁

使用 VarHandle 替代 volatile,实现更高效的内存可见性控制:

public class VarHandleSingleton {private static VarHandleSingleton instance;private static final VarHandle INSTANCE_HANDLE;static {try {INSTANCE_HANDLE = MethodHandles.lookup().findStaticVarHandle(VarHandleSingleton.class, "instance", VarHandleSingleton.class);} catch (ReflectiveOperationException e) {throw new Error(e);}}private VarHandleSingleton() {}public static VarHandleSingleton getInstance() {if (instance == null) {synchronized (VarHandleSingleton.class) {if (instance == null) {INSTANCE_HANDLE.setVolatile(new VarHandleSingleton());}}}return instance;}
}

五、Records(JDK14+)​​ 实现不可变单例

结合 Records 类型创建不可变单例(需配合工厂方法):

public record DatabaseConfig(String url, String user) {private static final DatabaseConfig INSTANCE = new DatabaseConfig("jdbc:mysql://localhost:3306/db", "admin");public static DatabaseConfig getInstance() {return INSTANCE;}
}

六、Sealed Classes(JDK17+)​​ 限制子类化

通过密封类(Sealed Classes)防止单例被继承破坏:

public sealed class Logger permits LoggerSingleton {// 密封类定义
}public final class LoggerSingleton extends Logger {private static final LoggerSingleton INSTANCE = new LoggerSingleton();private LoggerSingleton() {}public static LoggerSingleton getInstance() {return INSTANCE;}
}

关键对比与选型建议

实现方式适用场景优势限制条件
Lambda + Supplier需要简洁语法 + 延迟加载代码简练,减少样板代码JDK8+
接口静态方法简单饿汉式单例天然支持全局访问点需配合实现类
模块化高安全性要求场景防止反射攻击JDK9+
VarHandle高性能要求的双重校验锁比 volatile 更底层控制JDK9+
Records不可变配置类单例自动生成equals/hashCode/toStringJDK14+(预览特性)
Sealed Classes防止单例被子类化破坏编译期安全检查JDK17+

总结

虽然 JDK8 后单例模式的核心思想未变,但新特性提供了 ​更安全、更简洁的实现选择​:

  1. 优先选择 Records/枚举​:对不可变配置类,Records 提供天然线程安全和简洁性。
  2. 高并发场景用 VarHandle​:替代传统双重校验锁,减少内存屏障开销。
  3. 模块化增强防御​:结合模块系统限制反射访问,提升安全性。
  4. 避免过度设计​:简单场景仍推荐枚举或静态内部类实现。

相关文章:

  • 第 12 届蓝桥杯 C++ 青少组中 / 高级组省赛 2021 年 4 月 24 日真题(选择题)
  • GoogleTest:TEST_F
  • php artisan resetPass 执行密码重置失败的原因?php artisan resetPass是什么 如何使用?-优雅草卓伊凡
  • 基于C++、JsonCpp、Muduo库实现的分布式RPC通信框架
  • 安妮推广导航系统开心版多款主题网址推广赚钱软件推广变现一键统计免授权源码Annie
  • 【SpringBoot】Spring中事务的实现:声明式事务@Transactional、编程式事务
  • 基于RT-Thread的STM32开发第一讲——USART
  • Java学习手册:Spring Security 安全框架
  • [javaEE]网络编程
  • python设置word字体的方法
  • linux进程的复制和替换
  • Cherry Studio的MCP协议集成与应用实践:从本地工具到云端服务的智能交互
  • Spring AI:简化人工智能功能应用程序开发
  • 数字时代,如何为个人信息与隐私筑牢安全防线?
  • Linux系统安装方式+适合初学者的发行版本
  • Python项目源码63:病历管理系统1.0(tkinter+sqlite3+matplotlib)
  • 泰迪杯特等奖案例学习资料:基于边缘计算与多模态融合的温室传感器故障自诊断系统设计
  • BBR 之 ProbeRTT 新改
  • 基于随机森林的糖尿病预测模型研究应用(python)
  • 颠覆者DeepSeek:从技术解析到实战指南——开源大模型如何重塑AI生态
  • 特朗普公开“怼”库克:苹果不应在印度生产手机
  • 总奖金池百万!澎湃与七猫非虚构写作与现实题材征文大赛征稿启动
  • 南昌上饶领导干部任前公示:2人拟提名为县(市、区)长候选人
  • 定制基因编辑疗法治愈罕见遗传病患儿
  • 病重老人被要求亲自取钱在农业银行门口去世?株洲警方介入
  • 鄂州交警通报致1死2伤车祸:女子操作不当引发,已被刑拘