spring之Bean的循环依赖问题

news2024/11/18 11:32:59

文章目录

  • 一、Bean的循环依赖之Set注入模式下
    • 1、Husband类
    • 2、Wife类
    • 3、Spring配置文件
    • 4、测试类
    • 5、测试结果
    • 6、结论
  • 二、Bean的循环依赖之构造方法注入模式下
    • 1、Husband类
    • 2、Wife类
    • 3、Spring配置文件
    • 4、测试类
    • 5、运行结果
  • 三、Spring解决循环依赖的机理
    • 三级缓存(面试常问)


一、Bean的循环依赖之Set注入模式下

A对象中有B属性,B对象中有A属性。这就是循环依赖。我依赖你,你也依赖我。
比如:丈夫类Husband、妻子类Wife。Husband类中有Wife类的引用。Wife类中有Husband类的引用。

1、Husband类

husband类中有wife

注意:里边的toString方法对于wife这个属性使用了getName()避免陷入死循环

public class Husband {
    private String name;
    private Wife wife;

    public void setName(String name) {
        this.name = name;
    }

    public void setWife(Wife wife) {
        this.wife = wife;
    }

    public String getName() {
        return name;
    }

    @Override
    public String toString() {
        return "Husband{" +
                "name='" + name + '\'' +
                ", wife=" + wife.getName() +
                '}';
    }
}

2、Wife类

wife类中有husband

public class Wife {
    private String name;
    private Husband husband;

    @Override
    public String toString() {
        return "Wife{" +
                "name='" + name + '\'' +
                ", husband=" + husband.getName() +
                '}';
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setHusband(Husband husband) {
        this.husband = husband;
    }
}

3、Spring配置文件

配置两个bean

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="husband" class="com.bjpowernode.spring.bean.Husband" scope="singleton">
        <property name="name" value="小白"></property>
        <property name="wife" ref="wife"></property>
    </bean>

    <bean id="wife" class="com.bjpowernode.spring.bean.Wife" scope="singleton">
        <property name="name" value="小川"></property>
        <property name="husband" ref="husband"></property>
    </bean>
</beans>

4、测试类

    @Test
    public void testDeprndency(){
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring2.xml");
        Husband husband = applicationContext.getBean("husband", Husband.class);
        System.out.println(husband);

        Wife wife = applicationContext.getBean("wife", Wife.class);
        System.out.println(wife);
    }

5、测试结果

在这里插入图片描述

6、结论

singleotn + setter模式下的循环依赖 spring是没有任何问题的。
singleotn 表示在整个Spring容器当中是单例的,独一无二的。

singleotn + setter模式下的循环依赖 spring是如何应对的?

主要原因是在这种模式下,Spring对Bean的管理主要分为清晰的两个阶段:
第一个阶段:在Spring容器加载的时候,实例化Bean,只要其中任意一个Bean实例化之后,马上进行“曝光”【不等属性赋值就曝光】
第二个阶段:Bean“曝光”之后,再进行属性的赋值。
核心解决方案是:实例化对象和对象的属性赋值分为两个阶段来完成的。
只有在scope是singleton的情况下,Bean才会采取提前“曝光”的措施

prototy+ setter模式下的循环依赖 spring是会出现异常的

    <bean id="husband" class="com.bjpowernode.spring.bean.Husband" scope="prototype">
        <property name="name" value="小白"></property>
        <property name="wife" ref="wife"></property>
    </bean>

    <bean id="wife" class="com.bjpowernode.spring.bean.Wife" scope="prototype">
        <property name="name" value="小川"></property>
        <property name="husband" ref="husband"></property>
    </bean>

在这里插入图片描述
Bean的循环依赖出现问题:BeanCurrentlyInCreationException
注意:当两个Bean的scope都是prototype的时候,才会出现异常,如果其中任意一个是singleton,就不会出现异常

二、Bean的循环依赖之构造方法注入模式下

1、Husband类

去掉set方法,加入构造方法

public class Husband {
    private String name;
    private Wife wife;

    public Husband(String name, Wife wife) {
        this.name = name;
        this.wife = wife;
    }
    public String getName() {
        return name;
    }
    @Override
    public String toString() {
        return "Husband{" +
                "name='" + name + '\'' +
                ", wife=" + wife.getName() +
                '}';
    }
}

2、Wife类

去掉set方法,加入构造方法

public class Wife {
    private String name;
    private Husband husband;

    public Wife(String name, Husband husband) {
        this.name = name;
        this.husband = husband;
    }

    @Override
    public String toString() {
        return "Wife{" +
                "name='" + name + '\'' +
                ", husband=" + husband.getName() +
                '}';
    }

    public String getName() {
        return name;
    }
}

3、Spring配置文件

构造注入,这种循环依赖是否会出现问题?

<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="husband" class="com.bjpowernode.spring.bean.Husband" scope="singleton">
        <constructor-arg index="0" value="小白"></constructor-arg>
        <constructor-arg index="1" ref="wife"></constructor-arg>
    </bean>

    <bean id="wife" class="com.bjpowernode.spring.bean.Wife" scope="singleton">
        <constructor-arg index="0" value="小川"></constructor-arg>
        <constructor-arg index="1" ref="husband"></constructor-arg>
    </bean>
</beans>

4、测试类

    @Test
    public void testDeprndency(){
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring2.xml");
        Husband husband = applicationContext.getBean("husband", Husband.class);
        System.out.println(husband);

        Wife wife = applicationContext.getBean("wife", Wife.class);
        System.out.println(wife);
    }

5、运行结果

在这里插入图片描述
注意:基于构造注入的方式下产生的循环依赖也是无法解决的

Spring只能解决set+singleton模式下的循环依赖。

三、Spring解决循环依赖的机理

根本原因在于:这种方式可以做到将“实例化Bean”和“给Bean属性赋值”这两个动作分开去完成。

  • 实例化Bean的时候:调用无参数构造方法来完成,此时可以先不给属性赋值,可以提前将该Bean对象“曝光”给外界。
  • 给Bean对象属性赋值的时候:调用setter方法来完成。

两个步骤完全是可以分离开去完成的,并且不要求在同一时间点上完成。

也就是说Bean都是单例的,我们可以先把所有的单例Bean实例化出来,放到一个集合当中(缓存),所有的单例Bean全部实例化之后。我们再慢慢的调用setter方法给属性赋值,这样就解决了循环依赖的问题。

追源码:
双击shift: AbstractAutowireCapableBeanFactory
ctrl + f :doCreateBean 方法

debug:
在这里插入图片描述
继续往下走一步:
这个husband对象的name属性和wife属性是空的,但是这个对象已经创建出来了
在这里插入图片描述
走到下一个断点:
在这里插入图片描述
把这个单例对象缓存起来,具体看看怎么缓存的:

三级缓存(面试常问)

step into进去一个新的类:DefaultSingletonBeanRegistry

在这里插入图片描述

先说说这个类DefaultSingletonBeanRegistry中的三个比较重要的缓存:
private final Map<String, Object> singletonObjects ------ 一级缓存
private final Map<String, Object> earlySingletonObjects ------ 二级缓存
private final Map<String, ObjectFactory<?>> singletonFactories ------ 三级缓存

这三个缓存都是Map集合。Map集合的key存储的都是bean的name(bean id)。

一级缓存存储的是:完整的单例Bean对象。也就是说这个缓存中的Bean对象的属性都已经赋值了。是一个完整的Bean对象

二级缓存存储的是:早期的单例Bean对象。这个缓存中的单例Bean对象的属性没有赋值,只是一个早期的实例对象

三级缓存存储的是:单例工厂对象。这个里面存储了大量的“工厂对象”,每一个单例Bean对象都会对应一个单例工厂对象。这个集合中存储的是:创建该单例对象时对应的那个单例工厂对象

继续回到debug:
进来DefaultSingletonBeanRegistry这个类之后addSingletonFactory这个方法执行
在这里插入图片描述
继续执行到:
并没有把Bean对象存进去,是把创建Bean对象的工厂对象存放到map集合。(三级缓存)
往map对象存的这个动作就叫做“曝光”
在这里插入图片描述
“曝光”工厂之后会继续调用getSingleton方法
在这里插入图片描述
然后从一级缓存取对象,拿不到从二级缓存取,拿不到从三级缓存取,
三级缓存取工厂对象,获取这个Bean对象,再把Bean对象放到二级缓存。
在这里插入图片描述
最后执行
populateBean(beanName, mbd, instanceWrapper);才会给属性赋值


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/112967.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PyQt5 基本布局管理 及 信号槽机制

一&#xff1a;布局设计 & 信号槽机制 效果实现如下&#xff1a; 对于窗口整体设计左右布局 对于左边布局&#xff0c;包括有水平布局(用户信息 左上方一块)垂直布局(多个按钮 左下方一块) 对于右边布局&#xff0c;主要是窗口切换&#xff0c;通过按下左边布局的左下方侧按…

SQLAlchemy连接MySQL及记录的查询、更新、删除、多表关联查询

SQLAlchemy是Python的ORM库&#xff0c;支持多种数据库。 建立连接 连接MySQL要用到Engine&#xff0c;Engine集成了连接池pool和方言Dialect&#xff08;支持不通数据库的SQL语法&#xff09;&#xff0c;最后都统一成标准DBAPI。 from sqlalchemy import create_engine en…

TypeScript

现在说起TypeScript想必大家都不会陌生的&#xff0c;当初从碎片信息中了解TypeScript&#xff0c;我认为他的变量声明和Rust语言有几分相似&#xff0c;是一门比较严格的语言&#xff0c;今天正式的来学习他 JavaScript易学习&#xff0c;易用&#xff0c;以至于大多数人对于…

软件体系结构 思维导图

软件体系结构 思维导图 软件体系结构思维导图 源文件放在 GitHub 仓库 使用 Xmind 即可打开查看 课程评价 比较抽象和理论化&#xff0c;如果光看 PPT 肯定看不懂&#xff0c;得听课或者看视频 后面实验试图基于 SpringBoot 去实战教学&#xff0c;可惜没系统学过只能照搬…

Kafka Consumer开发

Kafka Consumer - 消费者 跟生产者一样&#xff0c;消费者也属于kafka的客户端&#xff0c;不过kafka消费者是从kafka读取数据的应用&#xff0c;侧重于读数据。一个或多个消费者订阅kafka集群中的topic&#xff0c;并从broker接收topic消息&#xff0c;从而进行业务处理。今天…

一种嵌入式项目的参数保存方案

设计背景 嵌入式项目中&#xff0c;为了保证系统的正常运转&#xff0c;通常需要保存一部分数据至非易失存储设备如flash中。此处提供了一种通用的方案用于快速在项目中集成参数保存功能&#xff0c;该方案有以下几点特征&#xff1a; 接口简便&#xff0c;方便快速集成以及使用…

东北大学2023分布式操作系统实验

1.实验目的 建立伪分布式&#xff08;有条件的可以建立分布式环境&#xff09;的Hadoop环境&#xff0c;并成功运行示例程序。 2.Hadoop简介 2.1 Hadoop项目基础结构 在其核心&#xff0c;Hadoop主要有两个层次&#xff0c;即&#xff1a; 加工/计算层(MapReduce)存储层(Ha…

Python pandas有几千个库函数,你用过几个?(1)

对Python的 pandas 库所有的内置元类、函数、子模块等全部浏览一遍&#xff0c;然后挑选一些重点学习一下。我安装的库版本号为1.3.5&#xff0c;如下&#xff1a; >>> import pandas as pd >>> pd.__version__ 1.3.5 >>> print(pd.__doc__)pandas…

C++ STL vector list set map容器循环通过迭代器删除元素注意事项

先说说写这篇博客的原因吧&#xff0c;同事转部门了&#xff0c;把他手头的工作交接给了我。他以前维护的一个模块&#xff0c;会将外部输入的数据缓存起来分段处理&#xff0c;处理完了就会清除缓存数据&#xff0c;最近出现了一个bug&#xff0c;缓存数据一直不清除&#xff…

【SpringMVC】非注解的处理器映射器和适配器

项目目录 1.导入的依赖 pom.xml <properties><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><maven.compiler.source>1.8</maven.compiler.source><maven.compiler.target>1.8</maven.compiler.target><…

【K3s】第2篇 一篇文章学习实践K3s部署安装

目录 1、docker安装 2、docker-compose安装 3、K3s安装 3.1 k3s与install.sh文件准备 3.2 k3s 安装步骤 4、查看k3s部署状态 1、docker安装 方式一 https://fanjufei.blog.csdn.net/article/details/123500511https://fanjufei.blog.csdn.net/article/details/123500511 …

12.24

接口测试 ​ <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-width…

Unity3D异步加载场景SceneManager.LoadSceneAsync()卡住,并不异步,获取process直接到0.9的问题

问题阐述&#xff1a; 一般来说&#xff0c;在加载场景的时候&#xff0c;会因为所加载资源的大小、复杂度、电脑配置等因素导致加载过程耗费一定的时间。虽然这个加载时间是不可避免的&#xff0c;但是在这一小段卡着的时间里如果就这样卡着的话会大大降低玩家体验。 所以很多…

(matlab编程基础)数组的基本操作

目录 1、数组寻址 2、数组元素删除 3、数组查找和排序 &#xff08;1&#xff09;数组查找 &#xff08;2&#xff09;数组排序 4、数组运算 5、数组操作函数 数组操作主要从以下5部分进行介绍:数组寻址、数组元素的删除、数组查找和排序、数组运算和数组操作函数。 1、…

PS CS6视频剪辑基本技巧(五)添加logo、动画和画中画

系列讲座导读 PS CS6视频剪辑基本技巧&#xff08;一&#xff09;CS6可以实现的视频剪辑功能 PS CS6视频剪辑基本技巧&#xff08;二&#xff09;视频剪接和添加图片 PS CS6视频剪辑基本技巧&#xff08;三&#xff09;添加声音和字幕 PS CS6视频剪辑基本技巧&#xff08;四&am…

RMQ - ST表

RMQ - ST表 1、RMQ 简介 RMQ (Range Minimum / Maximum Query) 问题是指&#xff1a;对于长度为 nnn 的数列 AAA&#xff0c;回答若干询问 (A,i,j)(A, i, j)(A,i,j) (1≤i,j≤n)(1≤i,j≤n)(1≤i,j≤n)&#xff0c;返回数列 A 中区间在 [i,j][i,j][i,j] 中的最小 (大) 值所在…

Vue2.0全面教程

Vue2.0 学习视频地址 文章目录Vue2.01.vue 简介1.1.什么是vue1.2.vue的两个特性1.2.1.数据驱动视图1.2.2.双向数据绑定1.3.MVVM概述1.4.MVVM工作原理2.vue的基本使用2.1.基本使用步骤2.2.基本代码与MVVM的对应关系3.vue的调试工具3.1.安装vue-devtool调试工具3.2.配置Chrome浏…

Axios(三)

目录 1.Axios的默认配置 2.Axios创建Ajax实例对象发送请求 3.Axios拦截器 4.Axios取消请求 5.Axios文件结构说明 6.Axios创建过程 7.Axios对象创建过程模拟实现 8.Axios发送请求过程详解 9.模拟实现Axios发送请求 1.Axios的默认配置 <!doctype html> <html …

QT系列第7节 标准对话框使用

QT编程中对话框作为最常用的窗口经常被使用&#xff0c;本节介绍一些QT常用的标准对话框使用。 目录 1.选择单文件 2.选择多文件 3.选择目录 4.文件存储 5.选择颜色 6.选择字体 7.输入字符换/整数/浮点数/条目 8.消息对话框 9.进度对话框 10.向导对话框 1.选择单文件…

【圣诞节限定】今天教你如何用Html+JS+CSS绘制3D动画圣诞树

一、前言 应CSDN的邀请&#xff0c;这次给大家展示一波&#xff0c;如何用H5技术绘制3D圣诞树。 二、创意名 只采用简单的HtmlJSCSS 技术绘制。 三、效果展示 圣诞树修过如下&#xff1a; 四、编码实现 将源码复制保存到html中打开即可。 <!DOCTYPE html> <html lang…