【Spring】循环依赖

目录标题

https://docs.spring.io/spring-framework/reference/

什么是循环依赖

多个Bean互相引用,形成环路

循环依赖场景

  • 原型Bean的循环依赖
  • 单例bean之构造注入的循环依赖
  • 单例bean之setter注入的循环依赖

前两者无法解决,最后一种可以通过Spring提供的三级缓存来进行实现。

Java SE 演示

java 复制代码
@Component
public class ServiceA {

    private ServiceB serviceB;

    public void setServiceB(ServiceB serviceB) {
        this.serviceB = serviceB;
        System.out.println("A 里面设置了B");
    }

    // public ServiceA(ServiceB serviceB) {
    //     this.serviceB = serviceB;
    // }
}
java 复制代码
@Component
public class ServiceB {

    private ServiceA serviceA;

    public void setServiceA(ServiceA serviceA) {
        this.serviceA = serviceA;
        System.out.println("B 里面设置了A");
    }

    // public ServiceB(ServiceA serviceA) {
    //     this.serviceA = serviceA;
    // }
}
java 复制代码
public class ClientDemo {
    public static void main(String[] args) {
        clientSet();
        // clientConstruct();
    }

    /**
     * setter注入
     */
    private static void clientSet() {
        //创建serviceA
        ServiceA serviceA = new ServiceA();

        //创建serviceB
        ServiceB serviceB = new ServiceB();

        //将serviceA注入到serviceB中
        serviceB.setServiceA(serviceA);

        //将serviceB注入到serviceA中
        serviceA.setServiceB(serviceB);
    }

    /**
     * 构造注入
     */
    private static void clientConstruct(){
        // new ServiceA(new ServiceB(new ServiceA(new ServiceB())));
    }
}

Spring 容器演示

xml 复制代码
<!--    <bean id="a" class="com.example.demo.circulardependency.spring.A" scope="singleton">-->
    <bean id="a" class="com.example.demo.circulardependency.spring.A" scope="prototype">
        <property name="b" ref="b"/>
    </bean>

<!--    <bean id="b" class="com.example.demo.circulardependency.spring.B" scope="singleton">-->
    <bean id="b" class="com.example.demo.circulardependency.spring.B" scope="prototype">
        <property name="a" ref="a"/>
    </bean>
java 复制代码
public class ClientSpringContainer {
    public static void main(String[] args) {
        sampleDemo();
    }

    /**
     * spring
     *
     * 2024/2/2 11:40
     */
    private static void sampleDemo() {
        /**
         * setter注入
         * 
         * 11:39:14.055 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Creating shared instance of singleton bean 'a'
         * ---A created success
         * 11:39:14.064 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Creating shared instance of singleton bean 'b'
         * ---B created success
         */
        /**
         * 构造注入
         * 
         * Exception in thread "main" org.springframework.beans.factory.BeanCreationException:
         *
         */
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        A a = context.getBean("a", A.class);
        B b = context.getBean("b", B.class);
    }
}

三级缓存

Spring循环依赖 - CSDN博客

核心知识

三级缓存

一级缓存 :Map<String, Object> singletonObjects,我愿称之为成品单例池,常说的 Spring 容器就是指它,我们获取单例 bean 就是在这里面获取的,存放已经经历了完整生命周期的Bean对象
二级缓存 :Map<String, Object> earlySingletonObjects,存放早期暴露出来的Bean对象,Bean的生命周期未结束(属性还未填充完整,可以认为是 半成品的 bean, 实例化但未初始化的Bean对象
三级缓存 :Map<String, ObiectFactory<?>> singletonFactories,存放可以生成Bean的工厂(FactoryBean),用于生产(创建)对象

java 复制代码
/** Cache of singleton objects: bean name to bean instance. */
// 一级缓存:singleton对象的缓存:bean名称 - bean实例。
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

/** Cache of singleton factories: bean name to ObjectFactory. */
// 三级缓存:单例工厂的缓存:bean名称 - ObjectFactory
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

/** Cache of early singleton objects: bean name to bean instance. */
// 二级缓存:早期singleton对象的缓存:bean名称 - bean实例。
private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);

四大方法

  1. getSingleton():从容器里面获得单例的bean,没有的话则会创建 bean
  2. doCreateBean():执行创建 bean 的操作(在 Spring 中以 do 开头的方法都是干实事的方法)
  3. populateBean():创建完 bean 之后,对 bean 的属性进行填充
  4. addSingleton():bean 初始化完成之后,添加到单例容器池中,下次执行 getSingleton() 方法时就能获取到

三级缓存中的迁移

  1. A创建过程中需要B,于是A将自己放到三级缓存里面,去实例化B
  2. B实例化的时候发现需要A,于是B先查一级缓存,没有,再查二级缓存,还是没有,再查三级缓存,找到了A,然后把三级缓存里面的这个A放到二级缓存里面,并删除三级缓存里面的A
  3. B顺利初始化完毕,将自己放到一级缓存里面(此时B里面的A依然是创建中状态),然后回来接着创建A,此时B已经创建结束,直接从一级缓存里面拿到B,然后完成创建,并将A自己放到一级缓存里面。

三级缓存源码分析

【Spring】三级缓存

源码思维导图

Spring三级缓存源代码执行图

源码图例

课前问题

还剩下三个:

  • 为什么不可以用二级缓存?这部分我在网上搜寻了一下,跟AOP的代理有关(由于目前我对AOP不熟,怕误导了大家,就先欠着)
  • 开发中解决循环依赖?欠着
  • 循环依赖遇上AOP?欠着



推荐阅读

相关推荐
喵叔哟4 分钟前
重构代码中引入外部方法和引入本地扩展的区别
java·开发语言·重构
尘浮生10 分钟前
Java项目实战II基于微信小程序的电影院买票选座系统(开发文档+数据库+源码)
java·开发语言·数据库·微信小程序·小程序·maven·intellij-idea
不是二师兄的八戒34 分钟前
本地 PHP 和 Java 开发环境 Docker 化与配置开机自启
java·docker·php
爱编程的小生1 小时前
Easyexcel(2-文件读取)
java·excel
带多刺的玫瑰1 小时前
Leecode刷题C语言之统计不是特殊数字的数字数量
java·c语言·算法
计算机毕设指导62 小时前
基于 SpringBoot 的作业管理系统【附源码】
java·vue.js·spring boot·后端·mysql·spring·intellij-idea
Gu Gu Study2 小时前
枚举与lambda表达式,枚举实现单例模式为什么是安全的,lambda表达式与函数式接口的小九九~
java·开发语言
Chris _data2 小时前
二叉树oj题解析
java·数据结构
牙牙7052 小时前
Centos7安装Jenkins脚本一键部署
java·servlet·jenkins
paopaokaka_luck2 小时前
[371]基于springboot的高校实习管理系统
java·spring boot·后端