Spring Boot的单例模式实现代码

 更新时间:2026年03月17日 09:23:21   作者:程序员飞鱼  
Spring Boot里的单例不是GoF意义上的全JVM 唯一对象 + 私有构造器,而是由 Spring 容器管理的每个 ApplicationContext 仅一个实例,本文介绍Spring Boot的单例模式是如何实现的,感兴趣的朋友跟随小编一起看看吧

Spring Boot 里的单例不是 GoF 意义上的全 JVM 唯一对象 + 私有构造器。

而是由 Spring 容器管理的每个 ApplicationContext 仅一个实例

其本质是 IOC 容器用缓存表保存已创建的 Bean,并在 getBean() 时复用。

默认就是单例

  • 任何 @Component / @Service / @Repository / @Controller,或 @Bean默认 scope=singleton
  • 单例是容器级别:多个 ApplicationContext(测试、子上下文)各有一份。
@Component // 等价于 @Scope("singleton")
public class UserService {}

容器如何保证只有一个

核心在 DefaultSingletonBeanFactory/DefaultSingletonBeanRegistry三级缓存与同步控制:

  • 一级缓存 singletonObjects:已完全创建好的单例。
  • 二级缓存 earlySingletonObjects:为解决循环依赖而“提前曝光”的半成品对象。
  • 三级缓存 singletonFactories:可生成早期引用的工厂(通常是 AOP 代理工厂)。

创建流程(简化):

  1. getBean(name) → 若在 singletonObjects 里,直接返回。
  2. 不在 → 标记“正在创建”,调用 createBean() 完成实例化、依赖注入、BeanPostProcessor 等。
  3. 初始化完成后放入 singletonObjects,清理二/三级缓存,返回单例。
  4. 全过程对同名 bean 的创建有同步锁,避免并发重复创建,并通过早期引用解决 A→B→A 的构造/注入级循环依赖

何时创建

  • 默认容器刷新时预创建(preInstantiateSingletons())。
  • 标注 @Lazy 或全局开启 spring.main.lazy-initialization=true 时,延迟到首次 getBean() 才创建,但仍是单例

线程安全与可见性

  • 创建阶段受容器同步保护,发布到 singletonObjects 前已完成依赖注入与后置处理,具备安全发布语义。
  • 单例是多线程共享:业务字段若是可变状态,需自行保证线程安全(不可变、同步、并发集合等)。

@Bean@Configuration 的细节

  • @Configuration 类会被 CGLIB 增强,其 @Bean 方法默认 proxyBeanMethods=true
    • 在同一配置类内部互相调用 @Bean 方法时,调用会被拦截并从容器取同一个单例,不是 new 新对象。
  • 若显式 proxyBeanMethods=false(Spring Boot 为提速常用),同类内直接方法调用将变成普通方法调用,可能得到新对象
  • 只要是通过容器 getBean() 拿的,仍是单例。

经验:配置类间若互调 @Bean 方法,保持默认 proxyBeanMethods=true 或避免直接互调。

常见误区

  • Spring 单例 = JVM 唯一 ❌:是每个容器一个。
    • 你起了两个 ApplicationContext 就有两份。
  • 用 static/单例写法再交给 Spring 管理 ❌:多此一举且易与容器生命周期冲突。
  • 循环依赖全能解 ❌:仅能解单例构造器以外的注入型循环依赖,构造器循环、原型 scope、带代理的某些场景仍会失败。

最小验证

@SpringBootTest
class DemoTest {
    @Autowired UserService a;
    @Autowired ApplicationContext ctx;

    @Test void sameInstance() {
        UserService b = ctx.getBean(UserService.class);
        assertSame(a, b); // 同一容器内是同一个实例
    }
}

小结

Spring Boot 的单例= 默认 scope + BeanFactory 单例缓存 + 同步创建 +(必要时)早期曝光

把唯一性交给容器,比手写 GoF 单例更易测、更可控,也能与 AOP、生命周期、条件化装配等机制无缝协作。

到此这篇关于Spring Boot的单例模式实现代码的文章就介绍到这了,更多相关Spring Boot单例模式内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Mybatis Plus 增删改查的实现(小白教程)

    Mybatis Plus 增删改查的实现(小白教程)

    本文主要介绍了Mybatis Plus 增删改查,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09
  • 详解Java运算中的取余

    详解Java运算中的取余

    这篇文章主要介绍了java运算中的取余,在java运算中,取余符号是 %,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-05-05
  • Java基础语法之对象的比较方式

    Java基础语法之对象的比较方式

    文章介绍了Java中自定义类类型的比较大小的三种方式:覆写基类的equals方法、基于Comparable接口的比较、基于Comparator接口自定义比较器的比较,每种方式都有其适用场景和局限性
    2026-02-02
  • Java File类的详解及简单实例

    Java File类的详解及简单实例

    这篇文章主要介绍了Java File类的详解及简单实例的相关资料,希望通过本文大家能够掌握这部分内容,需要的朋友可以参考下
    2017-09-09
  • Java Swing实现扫雷小游戏

    Java Swing实现扫雷小游戏

    这篇文章主要为大家详细介绍了Java Swing实现扫雷小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • 深入理解Java Optional使用实践记录

    深入理解Java Optional使用实践记录

    Java8引入Optional类,用于优雅处理可能为空的值,避免空指针异常,提供创建、转换、过滤方法,支持链式调用,但应避免作为字段类型或集合元素,以防性能损耗,本文给大家介绍Java Optional使用实践记录,感兴趣的朋友一起看看吧
    2025-06-06
  • Java数据库操作库DButils类的使用方法与实例详解

    Java数据库操作库DButils类的使用方法与实例详解

    这篇文章主要介绍了JDBC数据库操作库DButils类的使用方法详解,需要的朋友可以参考下
    2020-02-02
  • 解决IDEA maven 项目修改代码不生效,mvn clean、install后才生效

    解决IDEA maven 项目修改代码不生效,mvn clean、install后才生效

    这篇文章主要介绍了解决IDEA maven 项目修改代码不生效,mvn clean、install后才生效的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-09-09
  • IDEA下从零开始搭建SpringBoot工程的方法步骤

    IDEA下从零开始搭建SpringBoot工程的方法步骤

    这篇文章主要介绍了IDEA下从零开始搭建SpringBoot工程的方法步骤,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-01-01
  • 新手学习java反射基础

    新手学习java反射基础

    Java反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;并且能改变它的属性。下面我们来一起学习一下吧
    2019-06-06

最新评论