Spring解决循环依赖实际就是用了个递归

本文用简要代码事例解释了Spring如何能解决依赖循环,并不讨论实际源码涉及的诸多因数,比如为什么有3个集合。

重点:

1、用3个集合来解决循环依赖,需要清楚3个集合分别是存什么

2开始条件 :使用 Supplier 来延迟 Bean 的创建,确保只有在需要时才创建 Bean。看下

代码中private Map<String, Supplier<?>> singletonFactories = new HashMap<>();

3 、**查找顺序:**①先从singletonObjects 找,②找不到再从earlySingletonObjects找,③再找不到从singletonFactories创建并放到earlySingletonObjects中。假设这个方法叫getBean。

4结束条件:从singletonFactories创建的时候遇到依赖会调用getBean获取 ,这相当于递归过程 。当遇到依赖循环的时候,递归会在调用步骤②的时候结束。相当于结束条件。

模拟代码:

java 复制代码
import java.util.HashMap;
import java.util.Map;
import java.util.function.Supplier;

class A {
    private B b;

    public void setB(B b) {
        this.b = b;
    }

    @Override
    public String toString() {
        return "A{" +
                "b=" + (b != null ? "B" : "null") +
                '}';
    }
}

class B {
    private A a;

    public void setA(A a) {
        this.a = a;
    }

    @Override
    public String toString() {
        return "B{" +
                "a=" + (a != null ? "A" : "null") +
                '}';
    }
}

public class CircularDependencyDemo {

    private Map<String, Object> singletonObjects = new HashMap<>();
    private Map<String, Object> earlySingletonObjects = new HashMap<>();
    private Map<String, Supplier<?>> singletonFactories = new HashMap<>();

    public static void main(String[] args) {
        CircularDependencyDemo demo = new CircularDependencyDemo();
        demo.resolveCircularDependency();
    }

    public void resolveCircularDependency() {
        //创建 A 的工厂并放入 singletonFactories
        singletonFactories.put("a", () -> {
            A a = new A();
            earlySingletonObjects.put("a", a);
            //初始化 A 时发现需要 B
            a.setB(getBean("b"));
            return a;
        });

        //创建 B 的工厂并放入 singletonFactories
        singletonFactories.put("b", () -> {
            B b = new B();
            earlySingletonObjects.put("b", b);
            //初始化 B 时发现需要 A
            b.setA(getBean("a"));
            return b;
        });

        //初始化 A 和 B
        A a = getBean("a");
        B b = getBean("b");

        //完成初始化后将 A 和 B 移动到 singletonObjects
        singletonObjects.put("a", a);
        singletonObjects.put("b", b);

        System.out.println(singletonObjects.get("a"));
        System.out.println(singletonObjects.get("b"));
    }

    //查找顺序
    private <T> T getBean(String beanName) {
        if (singletonObjects.containsKey(beanName)) {
            return (T) singletonObjects.get(beanName);
        }

        if (earlySingletonObjects.containsKey(beanName)) {
            return (T) earlySingletonObjects.get(beanName);
        }

        if (singletonFactories.containsKey(beanName)) {
            Supplier<?> factory = singletonFactories.remove(beanName);
            T object = (T) factory.get();
            //这里隐藏着递归过程
            earlySingletonObjects.put(beanName, object);
            return object;
        }

        throw new RuntimeException("Bean not found: " + beanName);
    }
}

上面是基本思路,实际源码涉及过程要复杂很多:

实际递归调用并非简单的直接调getSingletion,而是...->doCreateBean->getSingletion

doCreateBean做了以下几件事情

  1. 实例化 Bean(通过构造函数或工厂方法)
  2. 提前暴露 ObjectFactory(用于解决循环依赖)
  3. 填充属性(依赖注入)
  4. 初始化(调用 init-method、Aware 接口、BeanPostProcessor 等)
  5. 注册为完整单例

AbstractAutowireCapableBeanFactory .doCreateBean

java 复制代码
protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args) {
    //1.实例化 Bean(调用构造函数)
    BeanWrapper instanceWrapper = createBeanInstance(beanName, mbd, args);
    final Object bean = instanceWrapper.getWrappedInstance();

    // ... 省略部分代码 ...

    //2.【关键】判断是否需要提前暴露 ObjectFactory(用于循环依赖)
    boolean earlySingletonExposure = (mbd.isSingleton() 
            && this.allowCircularReferences 
            && isSingletonCurrentlyInCreation(beanName));

    if (earlySingletonExposure) {
        //3.注册一个 ObjectFactory 到 singletonFactories
        addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
    }

    //4.填充属性(populateBean)→ 这里会触发依赖的 Bean 创建(如 A 需要 B)
    populateBean(beanName, mbd, instanceWrapper);

    //5.初始化(initializeBean)
    Object exposedObject = initializeBean(beanName, bean, mbd);

    //6.如果提前暴露了,现在要把完整的 Bean 放入 singletonObjects
    if (earlySingletonExposure) {
        Object earlySingletonReference = getSingleton(beanName, false);
        if (earlySingletonReference != null) {
            // 处理代理替换等逻辑...
        }
    }

    return exposedObject;
}

填充属性间接调用了doCreateBean

java 复制代码
AbstractAutowireCapableBeanFactory.populateBean()
  └─> AbstractAutowireCapableBeanFactory.applyPropertyValues()
        └─> BeanDefinitionValueResolver.resolveValueIfNecessary()
              └─> BeanDefinitionValueResolver.resolveReference()
                    └─> AbstractBeanFactory.resolveEmbeddedValue() // (非关键)
                    └─> **AbstractBeanFactory.getBean(String beanName)**
                          └─> AbstractBeanFactory.doGetBean()
                                └─> **createBean()**
                                      └─> **doCreateBean()**  ← 最终入口
相关推荐
神奇的程序员3 小时前
从已损坏的备份中拯救数据
运维·后端·前端工程化
Goldn.3 小时前
Java核心技术栈全景解析:从Web开发到AI融合
java· spring boot· 微服务· ai· jvm· maven· hibernate
oden3 小时前
AI服务商切换太麻烦?一个AI Gateway搞定监控、缓存和故障转移(成本降40%)
后端·openai·api
李慕婉学姐4 小时前
【开题答辩过程】以《基于Android的出租车运行监测系统设计与实现》为例,不知道这个选题怎么做的,不知道这个选题怎么开题答辩的可以进来看看
java·后端·vue
m0_740043734 小时前
SpringBoot05-配置文件-热加载/日志框架slf4j/接口文档工具Swagger/Knife4j
java·spring boot·后端·log4j
编织幻境的妖4 小时前
SQL查询连续登录用户方法详解
java·数据库·sql
未若君雅裁5 小时前
JVM面试篇总结
java·jvm·面试
kk哥88995 小时前
C++ 对象 核心介绍
java·jvm·c++
招风的黑耳5 小时前
我用SpringBoot撸了一个智慧水务监控平台
java·spring boot·后端
xunyan62345 小时前
面向对象(下)-接口的理解
java·开发语言