首页 > 编程语言 >Spring源码系列五:循环依赖

Spring源码系列五:循环依赖

时间:2024-08-10 12:28:04浏览次数:9  
标签:缓存 依赖 对象 Spring 半成品 源码 循环

更多请关注:https://t.zsxq.com/fhroW

文章目录


什么是循环依赖?
循环依赖是spring中的问题,普通的java项目不会有循环依赖。Spring中出现的循环依赖有三种情况:自我依赖、循环依赖、多组依赖

什么是循环依赖?

循环依赖是spring中的问题,普通的java项目不会有循环依赖。

// A依赖了B
class A{
	public B b;
}

// B依赖了A
class B{
	public A a;
}

以上代码的逻辑就会产生循环依赖,创建A时需要依赖B,创建B时又需要A,陷入了循环。
Spring中出现的循环依赖有三种情况:自我依赖、循环依赖、多组依赖。

解决思想

解决循环依赖的关键在于打破循环。如何打破循环呢?
spring中将A的半成品对象存入缓存(map)中,创建B时,如果没有A的成品对象,就使用A的半成品对象。以此来打破循环。

B依赖的半成品对象会在A创建完成后变成成品对象,因为指向同一块存储空间。

spring如何解决循环依赖

  1. 为什么需要三级缓存
    Spring中使用三级缓存来解决循环依赖。按照上文的思想,解决循环依赖只需要一个缓存即可,为什么Spring要使用三级缓存?
  • 为了遵从Spring的设计思想
  • 为了代码逻辑清晰

如果只使用一个缓存,对象实例化后就把未进行属性填充半成品对象存入缓存中。这样设计有两个问题:

  1. 这个缓存中既有半成品对象,也有成品对象,容易造成空指针,代码逻辑不清晰
  2. 如果对象进行了AOP,那么用户最终得到的是代理对象,但是缓存中存的是普通对象

解决第一个问题
再加一个缓存,分别存储成品对象和半成品对象。
存储成品对象即Spring中的第一级缓存: singletonObjects
存储半成品对象即Spring中的第二级缓存:earlySingletonObjects

解决第二个问题
提前进行AOP,AOP本来是在初始化后中进行的,如果出现了循环依赖,将AOP过程提前到实例化后。这样在实例 化后得到的半成品对象可能是普通对象也可能是代理对象。
实例化后处理AOP的这一段逻辑额也存在缓存中,即Spring中的第三级缓存:singletonFactoies
(可能因为存的是生成半成品对象的方法,所以名字以factory结尾)

为什么不把所有的Bean都在实例化后进行AOP?
因为这违背了Spring先创建所有的Bean,再进行初始化的设计思想。

  1. 使用三级缓存的流程
    Spring中使用三级缓存解决循环依赖的流程:
    在这里插入图片描述

在实例化后将一个方法引用(getEarlyBeanReference())存入第三级缓存,执行这个方法可以获得一个半成品对象,可能是普通对象,也可能是普通对象。
在填充属性时,一次从第一级、第二级、第三级缓存中获取对象,进行填充。

从第三级缓存中获取到半成品对象后,会将该对象从第三级缓存中移除,将半成品对象添加到第二级缓存,下次找的时候就能从第二级缓存中拿到了。
对象创建完成之后会注册对象,即将对象从第二级缓存中移除,添加到第一级缓存中。下次从第一级缓存就直接拿到了。

三级缓存

  1. 三级缓存分别是什么时候存入的?
    实例化后存入第三级缓存,从第三级缓存中获取到半成品对象后,存入第二级缓存,对象创建完成后存入第一级缓存。
  2. 三级缓存分别是什么时候使用的?
    填充属性时依次从三级缓存中寻找对应的Bean
  3. 三级缓存分别存的是什么?
  • 一级缓存:即单例池,存储成品对象
  • 二级缓存:存储半成品对象
  • 三级缓存:存储的是获取半成品的方法引用。可以说存的是工厂对象
  1. 三级缓存的意义?
    三个缓存的意义都是打破循环,解决循环依赖。为了逻辑清晰增加了第二级循环,为了处理AOP且不违背设计思想,增加了第三级循环。
    所以如果要说三级缓存各级缓存的意义,我觉得可以说:第三级缓存打破了循环,第二级循环和第一级循环使代码逻辑清晰。

补充

  1. 判断是否出现了循环依赖?
    在创建Bean时,将这个Bean存入一个Map中,表示正在创建,创建结束后移除。其他Bean在创建时,如果发现依赖了这个对象,且这个对象存在于Map中,则说明出现了循环依赖。

标签:缓存,依赖,对象,Spring,半成品,源码,循环
From: https://blog.csdn.net/qq_43081232/article/details/141088877

相关文章

  • 【Java毕设选题推荐】基于SpringBoot的springbootJAVA的邮件过滤系统设计与实现
    前言:我是IT源码社,从事计算机开发行业数年,专注Java领域,专业提供程序设计开发、源码分享、技术指导讲解、定制和毕业设计服务......
  • 【Java毕设选题推荐】基于SpringBoot的O2O生鲜食品订购小程序
    前言:我是IT源码社,从事计算机开发行业数年,专注Java领域,专业提供程序设计开发、源码分享、技术指导讲解、定制和毕业设计服务......
  • Spring Boot集成Redis
    目录1.Redis简介2.添加依赖3.项目配置4.使用RedisStringTestSetTestListTestHashTestZSetTest5.测试结果1.Redis简介Redis是一款基于内存的使用K-V结构存取数据的NoSQL非关系型数据库。Redis的主要作用就是缓存数据,通常,会将关系型数据库(例如MySQL等)中的数据读......
  • 毕业设计:基于Springboot的口腔医院微信小程序【代码+论文+PPT】
    全文内容包括:1、采用技术;2、系统功能;3、系统截图;4、配套内容。索取方式见文末微信号,欢迎关注收藏!一、采用技术语言:Java1.8框架:SpringBoot数据库:MySQL5.7、8.0开发工具:IntelliJIDEA旗舰版、微信开发工具其他:Maven3.8以上二、系统功能系统首页:展示口腔医院微信小程序的主......
  • Springboot+Mybatis实现最简单的增删改查
    1.创建一个SpringBoot项目 2.导入依赖勾选上web下面的SpringWeb和sql下面的MySQL和Mybatis依赖 然后等待依赖下载导入2.1 依赖导入后的页面3.创建需要的包和Mybatis配置文件 3.1在原来的com.mybatis包下创建了entity,controller,service,mapper四个包3.2在resource......
  • [网络] 关键类及源码分析:java.net包
    0序1java.net包概述2关键类及源码分析2.1java.net.InetAddresspublicstaticvoidmain(String[]args)throwsUnknownHostException{ //1.获取本机的InetAddress对象 InetAddresslocalHost=InetAddress.getLocalHost(); System.out.println(localHost);//LAPT......
  • Spring AI 更新:支持OpenAI的结构化输出,增强对JSON响应的支持
    就在昨晚,SpringAI发了个比较重要的更新。由于最近OpenAI推出了结构化输出的功能,可确保AI生成的响应严格遵守预定义的JSON模式。此功能显着提高了人工智能生成内容在现实应用中的可靠性和可用性。SpringAI紧随其后,现在也可以对OpenAI的结构化输出完美支持了。下图展示了本......
  • 【深入理解SpringCloud微服务】Ribbon源码解析
    【深入理解SpringCloud微服务】Ribbon源码解析Ribbon的原理RestTemplate中的拦截器链Ribbon的拦截器如何将拦截器放入到RestTemplate中Ribbon中的核心类LoadBalancerAutoConfigurationLoadBalancerInterceptorLoadBalancerClientILoadBalancerServerListIRuleIPingRibb......
  • OpenCV与AI深度学习 | 手把手教你用Python和OpenCV搭建一个半自动标注工具(详细步骤 +
    本文来源公众号“OpenCV与AI深度学习”,仅用于学术分享,侵权删,干货满满。原文链接:手把手教你用Python和OpenCV搭建一个半自动标注工具(详细步骤+源码)导 读    本文将手把手教你用Python和OpenCV搭建一个半自动标注工具(包含详细步骤+源码)。背景介绍    样本标......
  • 基于Springboot的仿淘宝系统(有报告)。Javaee项目,springboot项目。
    演示视频:基于Springboot的仿淘宝系统(有报告)。Javaee项目,springboot项目。项目介绍:采用M(model)V(view)C(controller)三层体系结构,通过Spring+SpringBoot+Mybatis+Vue+Maven来实现。MySQL数据库作为系统数据储存平台,实现了基于B/S结构的Web系统。界面简洁,操作简单。......