小红书二面:Spring Boot的单例模式是如何实现的?

文章内容收录到个人网站,方便阅读hardyfish.top/

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

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

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

默认就是单例

  • 任何 @Component / @Service / @Repository / @Controller,或 @Bean默认 scope=singleton
  • 单例是容器级别 :多个 ApplicationContext(测试、子上下文)各有一份。
java 复制代码
@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、带代理的某些场景仍会失败。

最小验证

java 复制代码
@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、生命周期、条件化装配等机制无缝协作。

相关推荐
jiayong2317 分钟前
前端面试题库 - JavaScript核心基础篇
前端·javascript·面试
JAVA面经实录91718 分钟前
Java多线程并发高频面试100题(完整版·含答案·背诵版)
java·开发语言·面试
XiYang-DING22 分钟前
【Java EE】TCP—流量控制和拥塞控制
java·tcp/ip·java-ee
用户67570498850232 分钟前
Celery 太重了?这可能是你一直在找的 asyncio 任务队列
后端·python·消息队列
Cloud_Shy61833 分钟前
Python 数据分析基础入门:《Excel Python:飞速搞定数据分析与处理》学习笔记系列(第十一章 Python 包跟踪器 下篇)
前端·后端·python·数据分析·excel
BIG_PEI41 分钟前
检查并安装Redis
java
大貔貅喝啤酒44 分钟前
基于Windows下载安装Android Studio 3.3.2版本教程(2026详细图文版)
android·java·windows·android studio
奋斗的小方1 小时前
Java基础篇09:项目实战
java·开发语言
海兰1 小时前
【第21篇-续】graph-Stream-Node改造为适配openAI模型示例
java·人工智能·spring boot·spring·spring ai
vKd0Ff21L1 小时前
如何在Dev-C++中设置TDM-GCC为默认编译器第九十一篇
java·jvm·c++