Spring循环依赖及三级缓存

简介: Spring循环依赖及三级缓存源码解析

循环依赖

类A和类B,A实例化的时候需要B的实例,B实例化的时候需要A的实例,这样就进入了死循环。

@Data
@Component
public class Org{
    private final Role role;
    public Org(Role role){
        this.role=role;
    }
}
@Data
@Component
public class Role{
    private final Org org;
    public Role(Org org){
        this.org=org;
    }
}

这就是spring中典型额构造器注入方式。此时启动会出错。

而如果改一下代码,把构造器注入改为属性注入(@Autowired,@Resource),就不会报错,两个bean都实例化成功了。说明spring框架有解决循环依赖的过程。

三级缓存

spring创建bean的过程分为三步:

1.实例化,对应方法AbstractAutowireCapableBeanFactory中的createBeanInstance方法,简单理解就是new了一个对象。

2.属性注入,对应方法AbstractAutowireCapableBeanFactory中的populateBean方法,为实例化中new出来的对象填充属性和注入依赖。

3.初始化,对应方法AbstractAutowireCapableBeanFactory的initializeBean,执行aware接口中的方法,初始化方法,完成aop代理

从单例Bean的初始化来看,主要可能发生循环依赖的环节就在第二部populate。值得注意的是,给予勾到方法注入的方式,其实是将第一步和第二部同时进行,因此马上就抛出错误。而spring通过属性注入的方法,是否有其他特殊处理呢,这时候就提到了三级缓存:

private final Map<String,Object> singletonObject = new ConcurrentHashMap<>(256);
private final Map<String,Object> earlySingletonObjects = new ConcurrentHashMap<>(16);
private final Map<String,ObjectFactory<?>> singletonFactories = new ConcurrentHashMap<>(16);

singletonObject:第一级缓存,存放可用的完全初始化,成品bean。

earlySingletonObjects:第二级缓存,存放半成品bean,指的是已创建对象,但是未注入属性和初始化,用以解决循环依赖。

singletonFactories:第三级缓存,存的事bean工厂对象,用来生成半成品的bean并放入到二级缓存中。用以解决循环依赖。如果bean存在AOP的话,返回的就是AOP的代理对象。

核心方法:getSingleton

我们在获取bean实例的时候,其实是先从三级缓存中获取:

protected Object getSingleton(String beanName, boolean allowEarlyReference) {
//查询缓存中是否有创建好的单例
   Object singletonObject = this.singletonObjects.get(beanName);
  //isSingletonCurrentlyInCreation 判断对应的单例对象是否在创建中
  //如果不存在,判断是否正在创建中
  //当单例对象没有被初始化完全(例如A定义的构造函数依赖了B对象,得先去创建B对象,或者在populatebean过程中依赖了B对象,得先去创建B对象,此时A处于创建中)
   if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
  synchronized (this.singletonObjects) {
//从earlySingletonObjects冲查询是否有early缓存
     singletonObject = this.earlySingletonObjects.get(beanName);
    //allowEarlyReference 是否允许从singletonFactories中通过getObject拿到对象
    //early缓存也不存在,且允许early使用
     if (singletonObject == null && allowEarlyReference) {
    //从单例工厂Map里查询beanName
        ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
        if (singletonFactory != null) {
    //singletonFactory存在,则调用getObject方法拿到单例对象
           singletonObject = singletonFactory.getObject();
          //将单例对象添加到early缓存中 this.earlySingletonObjects.put(beanName, singletonObject);
          //移除单例工厂中对应的singletonFactory this.singletonFactories.remove(beanName);
        }
     }
  }
   }
   return (singletonObject != NULL_OBJECT ? singletonObject : null);}

三级缓存中一级一级的找匹配的bean,直到最后一级缓存,通过匹配beanName和ObjectFactory来获取Bean,namesingleFactories何时放入了可以通过getObject获得bean对象的ObjectFactory呢?

核心方法:doCreateBean

Bean的实例化,实际执行的源码是AbstractAutowireCapableBeanFactory类的doCreateBean方法;

方法中的BeanWrapper接口,他是bean的包裹类,即在内部中将会保存该bean的实例,提供其他一些扩展功能。同时,BeanWrapper接口还继承了PropertyAccessor,propertyEditorRegistry,TypeConverter,ConfigurationPropertyAccessor接口,所以他还提供了访问bean的属性值,属性编辑器注册,类型转换等功能。

回顾一下bean实例化过程:

ResourceLoader加载配置信息-》BeanDefinitionReader读取bean标签,并将bean标签的属性转换为BeanDefinition对应的属性,并注册到BeanDefinitionRegistry注册表中-》容器扫描注册表,通过反射机制获取BeanFactoryPostPropcessor类型的工厂处理器,并用这个工厂后处理器对BeanDefinition进行加工-》根据处理过的BeanDefinition,实例化Bean,然后BeanWrapper结合BeanDefitionRegistry和PropertyEditorRegistry对Bean的属性赋值。

思考总结:

1.多例循环依赖可以解决吗?

多例是每次创建对象都会调用doGetBean方法,根本没有使用一二三级缓存,所以多例是无法解决的。

2.构造器。setter注入的方式的循环依赖可以决绝吗?

类A和类B

均采用setter方法注入:可以解决

均采用构造器注入:不可解决

A注入B为setter,B注入A为构造器:可以解决

B注入A为setter,A注入B为构造器:不可决绝

出现循环依赖,只能先用AOP给bean先创建代理,三级缓存singleFactory的目的就是,暴露ObjectFactory而完成AOP代理,对象工厂清楚如何创建对象的AOP代理,但是不会立马创建,而是到合适的时机进行AOP代理对象的创建。

二级缓存存在的目的之一就是保证对象只有一次AOP代理,当调用三级缓存getObject()方法返回的对象会存入二级缓存,这样,当接下来的依赖者调用的时候,会先判断二级缓存是否存在,日过存在直接返回。

总结:

一级缓存为单例池,二级缓存为早期曝光对象,三级缓存为早期曝光对象工厂。

当A、B两类发生循环引用后,将自己提早曝光(加入三级缓存),如果A初始AOP代理,该工厂对象返回的事被代理的对象,若未被代理,返回对象本身。当A进行属性注入时,经过之前实例化步骤,此时轮到B属性注入,调用getBean(a)获取A对象,由于A处理正在创建集合中,此时也发生了循环依赖,所以可以从三级换窜获取对象工厂(如果A被AOP代理,此时返回就是代理对象),并把对象放到二级缓存中,这样保证A只经过一次AOP代理。接下来,B走完Spring生命周期流程,并放入单例池中。当B创建完后,会将B注入A,A走完Spring生命周期流程。到此,循环依赖结束。

相关文章
|
1月前
|
缓存 Java 开发工具
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
三级缓存是Spring框架里,一个经典的技术点,它很好地解决了循环依赖的问题,也是很多面试中会被问到的问题,本文从源码入手,详细剖析Spring三级缓存的来龙去脉。
134 24
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
|
16天前
|
缓存 Java Spring
源码解读:Spring如何解决构造器注入的循环依赖?
本文详细探讨了Spring框架中的循环依赖问题,包括构造器注入和字段注入两种情况,并重点分析了构造器注入循环依赖的解决方案。文章通过具体示例展示了循环依赖的错误信息及常见场景,提出了三种解决方法:重构代码、使用字段依赖注入以及使用`@Lazy`注解。其中,`@Lazy`注解通过延迟初始化和动态代理机制有效解决了循环依赖问题。作者建议优先使用`@Lazy`注解,并提供了详细的源码解析和调试截图,帮助读者深入理解其实现机制。
17 1
|
27天前
|
缓存 Java Spring
手写Spring Ioc 循环依赖底层源码剖析
在Spring框架中,IoC(控制反转)是一个核心特性,它通过依赖注入(DI)实现了对象间的解耦。然而,在实际开发中,循环依赖是一个常见的问题。
33 4
|
29天前
|
存储 缓存 Java
在Spring Boot中使用缓存的技术解析
通过利用Spring Boot中的缓存支持,开发者可以轻松地实现高效和可扩展的缓存策略,进而提升应用的性能和用户体验。Spring Boot的声明式缓存抽象和对多种缓存技术的支持,使得集成和使用缓存变得前所未有的简单。无论是在开发新应用还是优化现有应用,合理地使用缓存都是提高性能的有效手段。
27 1
|
2月前
|
存储 缓存 Java
面试问Spring循环依赖?今天通过代码调试让你记住
该文章讨论了Spring框架中循环依赖的概念,并通过代码示例帮助读者理解这一概念。
面试问Spring循环依赖?今天通过代码调试让你记住
|
2月前
|
缓存 NoSQL Java
SpringBoot的三种缓存技术(Spring Cache、Layering Cache 框架、Alibaba JetCache 框架)
Spring Cache 是 Spring 提供的简易缓存方案,支持本地与 Redis 缓存。通过添加 `spring-boot-starter-data-redis` 和 `spring-boot-starter-cache` 依赖,并使用 `@EnableCaching` 开启缓存功能。JetCache 由阿里开源,功能更丰富,支持多级缓存和异步 API,通过引入 `jetcache-starter-redis` 依赖并配置 YAML 文件启用。Layering Cache 则提供分层缓存机制,需引入 `layering-cache-starter` 依赖并使用特定注解实现缓存逻辑。
582 1
SpringBoot的三种缓存技术(Spring Cache、Layering Cache 框架、Alibaba JetCache 框架)
|
2月前
|
缓存 Java Spring
spring如何解决循环依赖
Spring框架处理循环依赖分为构造器循环依赖与setter循环依赖两种情况。构造器循环依赖不可解决,Spring会在检测到此类依赖时抛出`BeanCurrentlyInCreationException`异常。setter循环依赖则通过缓存机制解决:利用三级缓存系统,其中一级缓存`singletonObjects`存放已完成的单例Bean;二级缓存`earlySingletonObjects`存放实例化但未完成属性注入的Bean;三级缓存`singletonFactories`存放创建这些半成品Bean的工厂。
|
2月前
|
缓存 Java Spring
Spring缓存实践指南:从入门到精通的全方位攻略!
【8月更文挑战第31天】在现代Web应用开发中,性能优化至关重要。Spring框架提供的缓存机制可以帮助开发者轻松实现数据缓存,提升应用响应速度并减少服务器负载。通过简单的配置和注解,如`@Cacheable`、`@CachePut`和`@CacheEvict`,可以将缓存功能无缝集成到Spring应用中。例如,在配置文件中启用缓存支持并通过`@Cacheable`注解标记方法即可实现缓存。此外,合理设计缓存策略也很重要,需考虑数据变动频率及缓存大小等因素。总之,Spring缓存机制为提升应用性能提供了一种简便快捷的方式。
44 0
|
2月前
|
缓存 NoSQL Java
惊!Spring Boot遇上Redis,竟开启了一场缓存实战的革命!
【8月更文挑战第29天】在互联网时代,数据的高速读写至关重要。Spring Boot凭借简洁高效的特点广受开发者喜爱,而Redis作为高性能内存数据库,在缓存和消息队列领域表现出色。本文通过电商平台商品推荐系统的实战案例,详细介绍如何在Spring Boot项目中整合Redis,提升系统响应速度和用户体验。
58 0
|
2月前
|
Java Spring 容器
循环依赖难破解?Spring Boot神秘武器@RequiredArgsConstructor与@Lazy大显神通!
【8月更文挑战第29天】在Spring Boot应用中,循环依赖是一个常见问题。当两个或多个Bean相互依赖形成闭环时,Spring容器会陷入死循环。本文通过对比@RequiredArgsConstructor和@Lazy注解,探讨它们如何解决循环依赖问题。**@RequiredArgsConstructor**:通过Lombok生成包含final字段的构造函数,优先通过构造函数注入依赖,简化代码但可能导致构造函数复杂。**@Lazy**:延迟Bean的初始化,直到首次使用,打破创建顺序依赖,增加灵活性但可能影响性能。根据具体场景选择合适方案可有效解决循环依赖问题。
60 0