【Spring】循环依赖

这篇具有很好参考价值的文章主要介绍了【Spring】循环依赖。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

  • 循环依赖是什么?
  • 循环依赖的场景有哪一些?
  • 三级缓存分别是什么?三个Map有什么异同?
  • 三级缓存是如何解决循环依赖的?
  • 为什么要使用三级缓存?为什么不可以用二级缓存?
  • 为什么构造器循环依赖、原型Bean循环依赖无法用三级缓存解决?
  • 看过 Spring源码吗?一般我们说的 Spring容器是什么?
  • 如何检测是否存在循环依赖?实际开发中见过循环依赖的异常吗?
  • 如果循环依赖的时候,所有类又都需要 Spring AOP自动代理,那Spring如何提前曝光?曝光的是 原始bean 还是 代理后的bean ?

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

【Spring】循环依赖,Spring,源码分析,spring,java

什么是循环依赖

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

循环依赖场景

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

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

Java SE 演示

@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;
    // }
}
@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;
    // }
}
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 容器演示

<!--    <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>
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),用于生产(创建)对象

/** 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三级缓存源代码执行图

源码图例

【Spring】循环依赖,Spring,源码分析,spring,java

课前问题

【Spring】循环依赖,Spring,源码分析,spring,java
还剩下三个:

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

【Spring】循环依赖,Spring,源码分析,spring,java
【Spring】循环依赖,Spring,源码分析,spring,java
【Spring】循环依赖,Spring,源码分析,spring,java

【Spring】循环依赖,Spring,源码分析,spring,java

【Spring】循环依赖,Spring,源码分析,spring,java

【Spring】循环依赖,Spring,源码分析,spring,java文章来源地址https://www.toymoban.com/news/detail-832768.html

推荐阅读

  • Spring源码最难问题《当Spring AOP遇上循环依赖》_循环依赖aop在那个阶段-CSDN博客
  • spring 循环依赖以及解决方案(吊打面试官)_循环依赖解决方案-CSDN博客
  • spring循环依赖-CSDN博客

到了这里,关于【Spring】循环依赖的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • spring 的循环依赖以及spring为什么要用三级缓存解决循环依赖

            bean的生命周期         这里简单过一下 class -无参构造 -普通对象 -依赖注入(对加了autowire等的属性赋值) -初始化前-初始化 -初始化后(aop) -放入单例池的map(一级缓存) -bean对象 这里提一点单例bean单例bean 其实就是用mapbeanName,Bean对象创建的,多例bean就不

    2024年02月15日
    浏览(53)
  • 【Spring】Spring循环依赖(超重要!!)

    目录 什么是循环依赖问题 循环依赖具体是怎么解决的 具体的解决步骤: 通俗实例:       严谨的循环依赖解决图例 为什么使用的是三级缓存,二级缓存不够用吗?         Spring的循环依赖是指在Bean之间存在相互依赖关系,形成一个闭环的情况。简单来说,Bean A依赖于

    2024年02月11日
    浏览(32)
  • 【Spring】Spring循环依赖的处理

    循环依赖是指两个或多个组件之间相互依赖,形成一个闭环,从而导致这些组件无法正确地被初始化或加载。这种情况可能会在软件开发中引起问题,因为循环依赖会导致初始化顺序混乱,组件之间的关系变得复杂,甚至可能引发死锁或其他不稳定行为。 在编程中,循环依赖

    2024年02月11日
    浏览(38)
  • 【Spring】Spring的循环依赖以及解决方案

    ​ Spring循环依赖指的是两个或多个Bean之间相互依赖,形成一个环状依赖的情况。通俗的说,就是A依赖B,B依赖C,C依赖A,这样就形成了一个循环依赖的环。 ​ Spring循环依赖通常会导致Bean无法正确地被实例化,从而导致应用程序无法正常启动或者出现异常。因此,Spring循环

    2024年04月25日
    浏览(35)
  • 【Spring】循环依赖

    循环依赖是什么? 循环依赖的场景有哪一些? 三级缓存分别是什么?三个Map有什么异同? 三级缓存是如何解决循环依赖的? 为什么要使用三级缓存?为什么不可以用二级缓存? 为什么构造器循环依赖、原型Bean循环依赖无法用三级缓存解决? 看过 Spring源码吗?一般我们说

    2024年02月21日
    浏览(35)
  • Spring解决循环依赖问题

    例如,就是A对象依赖了B对象,B对象依赖了A对象。(下面的代码属于 属性的循环依赖 ,也就是初始化阶段的循环依赖,区别与底下 构造器的循环依赖 ) 问题来了: A Bean创建 —— 依赖了 B 属性 ——  触发 B Bean创建 ——  B 依赖了 A 属性 ——  需要 A Bean(但A Bean还在创建

    2024年02月12日
    浏览(39)
  • Spring解决循环依赖

    目录 什么是spring循环依赖 什么情况下循环依赖可以被处理? spring 如何解决循环依赖 创建A这个Bean的流程 答疑 疑问:在给B注入的时候为什么要注入一个代理对象? 初始化的时候是对A对象本身进行初始化,而容器中以及注入到B中的都是代理对象,这样不会有问题吗? 三级

    2024年02月22日
    浏览(46)
  • Spring循环依赖处理

    循环依赖是指两个或多个组件之间相互依赖,形成一个闭环,从而导致这些组件无法正确地被初始化或加载。这种情况可能会在软件开发中引起问题,因为循环依赖会导致初始化顺序混乱,组件之间的关系变得复杂,甚至可能引发死锁或其他不稳定行为。 在编程中,循环依赖

    2024年02月07日
    浏览(36)
  • Spring如何解决循环依赖问题

    循环依赖问题在Spring中主要有三种情况: (1)通过构造方法进行依赖注入时产生的循环依赖问题。 (2)通过setter方法进行依赖注入且是在多例(原型)模式下产生的循环依赖问题。 (3)通过setter方法进行依赖注入且是在单例模式下产生的循环依赖问题。 在Spring中,只有第

    2024年02月06日
    浏览(45)
  • Spring怎么解决循环依赖问题?

    循环依赖是指 一个或多个对象之间存在直接或间接的依赖关系,这种依赖关系构成一个环形调用 , 举个例子 : A 依赖B , B依赖C , C依赖A , 这样就形成了循环依赖;   ①构造器的循环依赖:这种依赖spring是处理不了的,直接拋出BeanCurrentlyInCreationException异常。 ②单例模式下的se

    2024年02月08日
    浏览(50)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包