三级缓存与循环依赖

1. 三级缓存的职责划分

  • 一级缓存 singletonObjects:存放完全初始化完成的单例 Bean。
  • 二级缓存 earlySingletonObjects:存放提前暴露的早期引用(可能是原始对象,也可能是代理对象)。
  • 三级缓存 singletonFactories:存放 ObjectFactory,按需生成早期引用。

2. 为什么需要三级缓存

  • 如果只有一级缓存,循环依赖场景下无法提前拿到对象,注入会失败。
  • 如果只有二级缓存(直接放对象),很难优雅支持 AOP 等"可能需要提前生成代理"的场景。
  • 三级缓存通过工厂延迟创建早期引用:既能解决循环依赖,也能在需要时生成代理。

3. 循环依赖如何被解除(A -> B -> A

  • 创建 A 后先把 A 的 ObjectFactory 放入三级缓存。
  • A 依赖 B,转而创建 B;B 依赖 A 时,按顺序查一级、二级、三级。
  • 三级命中后调用工厂拿到 A 的早期引用,并放入二级缓存,再注入给 B。
  • B 完成后回到 A,A 注入 B 并完成初始化,最后进入一级缓存。

4. 动态代理(AOP)介入时发生什么

  • 关键点在 getEarlyBeanReference:当 B 反向依赖 A 时,A 的早期引用可被自动代理创建器包装成代理对象。
  • 因此 B 注入到的 A 往往是代理对象
  • A 自身后续初始化结束后,容器会对齐"最终暴露对象",确保一级缓存中是同一个代理实例,避免原始对象与代理对象不一致。

5. 实现与验证要点(本次补充)

  • 新增了可观测 Advice(计数器)用于验证代理拦截是否实际生效。

  • 新增了循环依赖 + 自动代理 XML 配置及测试。

  • 关键断言:

    • b.getA() == a(注入对象与容器最终对象一致)
    • a 为代理对象(例如 CGLIB 代理类名包含 $$
    • 调用切点方法后 Advice 计数增加。

时序图 1(中文):三级缓存解决循环依赖(无代理)

sequenceDiagram autonumber participant BF as Bean工厂 participant C1 as 一级缓存 participant C2 as 二级缓存 participant C3 as 三级缓存 participant A as BeanA participant B as BeanB BF->>BF: 获取Bean A BF->>A: 实例化A BF->>C3: 放入A的对象工厂 BF->>BF: A填充属性 发现依赖B BF->>BF: 获取Bean B BF->>B: 实例化B BF->>C3: 放入B的对象工厂 BF->>BF: B填充属性 发现依赖A BF->>C1: 查询A C1-->>BF: 未命中 BF->>C2: 查询A C2-->>BF: 未命中 BF->>C3: 查询A的对象工厂 C3-->>BF: 命中对象工厂 BF->>C3: 调用工厂创建A早期引用 BF->>C2: 放入A早期引用 BF->>C3: 删除A对象工厂 BF->>B: 将A早期引用注入B BF->>BF: B初始化完成 BF->>C1: 放入单例B BF->>A: 将B注入A BF->>BF: A初始化完成 BF->>C1: 放入单例A

时序图 2(英文):Three-level Cache Resolves Circular Dependency (No Proxy)

sequenceDiagram autonumber participant BF as BeanFactory participant C1 as Level1Cache participant C2 as Level2Cache participant C3 as Level3Cache participant A as BeanA participant B as BeanB BF->>BF: getBean A BF->>A: instantiate A BF->>C3: put factory for A BF->>BF: populate A, needs B BF->>BF: getBean B BF->>B: instantiate B BF->>C3: put factory for B BF->>BF: populate B, needs A BF->>C1: lookup A C1-->>BF: miss BF->>C2: lookup A C2-->>BF: miss BF->>C3: lookup factory for A C3-->>BF: factory found BF->>C3: call factory to get early A BF->>C2: put early A BF->>C3: remove factory for A BF->>B: inject early A into B BF->>BF: finish B initialization BF->>C1: put singleton B BF->>A: inject B into A BF->>BF: finish A initialization BF->>C1: put singleton A

时序图 3(中文):循环依赖 + 动态代理(A 被代理)

sequenceDiagram autonumber participant BF as Bean工厂 participant C1 as 一级缓存 participant C2 as 二级缓存 participant C3 as 三级缓存 participant APC as 自动代理创建器 participant A0 as A原始对象 participant AP as A代理对象 participant B as BeanB BF->>BF: 获取Bean A BF->>A0: 实例化A原始对象 BF->>C3: 放入A的对象工厂 BF->>BF: A填充属性 发现依赖B BF->>BF: 获取Bean B BF->>B: 实例化B BF->>C3: 放入B的对象工厂 BF->>BF: B填充属性 发现依赖A BF->>C1: 查询A C1-->>BF: 未命中 BF->>C2: 查询A C2-->>BF: 未命中 BF->>C3: 查询A对象工厂 C3-->>BF: 命中对象工厂 BF->>APC: 获取A早期引用 APC-->>BF: 返回A代理对象 BF->>C2: 放入A代理对象 BF->>C3: 删除A对象工厂 BF->>B: 注入A代理对象 BF->>BF: B初始化完成 BF->>C1: 放入单例B BF->>A0: 注入B并完成初始化 BF->>C2: 查询A早期引用 C2-->>BF: 返回A代理对象 BF->>C1: 放入单例A代理对象

时序图 4(英文):Circular Dependency with Dynamic Proxy (A is Proxied)

sequenceDiagram autonumber participant BF as BeanFactory participant C1 as Level1Cache participant C2 as Level2Cache participant C3 as Level3Cache participant APC as AutoProxyCreator participant A0 as A_RawObject participant AP as A_ProxyObject participant B as BeanB BF->>BF: getBean A BF->>A0: instantiate raw A BF->>C3: put factory for A BF->>BF: populate A properties, needs B BF->>BF: getBean B BF->>B: instantiate B BF->>C3: put factory for B BF->>BF: populate B properties, needs A BF->>C1: lookup A C1-->>BF: miss BF->>C2: lookup A C2-->>BF: miss BF->>C3: lookup factory for A C3-->>BF: factory found BF->>APC: get early reference for A APC-->>BF: return proxy A BF->>C2: put proxy A BF->>C3: remove factory for A BF->>B: inject proxy A BF->>BF: finish B initialization BF->>C1: put singleton B BF->>A0: inject B and initialize A BF->>C2: lookup early A reference C2-->>BF: return proxy A BF->>C1: put singleton proxy A
相关推荐
tongxh4231 小时前
Spring Boot 3.X:Unable to connect to Redis错误记录
spring boot·redis·后端
回家路上绕了弯1 小时前
Claude Superpower 全攻略:解锁 AI 自动开发模式,告别盲目编码
后端
kree2 小时前
Flowable 深度解析:现代企业级工作流引擎的核心与实践
后端
NCIN EXPE2 小时前
SpringBoot Test详解
spring boot·后端·log4j
2601_949194262 小时前
springboot之集成Elasticsearch
spring boot·后端·elasticsearch
splage3 小时前
Spring Cloud Data Flow 简介
后端·spring·spring cloud
woniu_maggie3 小时前
SAP RESTful 接口服务发布教程
后端
用户279420831323 小时前
临时解决 Mac SSH 客户端与服务器算法不匹配问题
后端
小锋java12343 小时前
LangChain4j 来了,Java AI智能体开发再次起飞。。。
java·人工智能·后端