spring的三级缓存,以及循环依赖的形成和解决(详细)

简介: spring的三级缓存,以及循环依赖的形成和解决(详细)


Spring循环依赖

当两个或更多个Bean之间相互依赖时,就会出现Spring循环依赖的问题。这意味着,每个Bean都需要其他Bean才能被创建,而其他Bean又需要该Bean才能被创建。这种情况下,Spring IoC容器会抛出一个异常,告诉你存在循环依赖。

什么是循环依赖

循环依赖是指两个或多个Bean相互依赖,导致Spring IoC容器无法正确创建它们。通常,循环依赖会导致无限递归,最终导致堆栈溢出。下面是一个简单的示例:

public class A {
    private B b;
    public A(B b) {
        this.b = b;
    }
}
public class B {
    private A a;
    public B(A a) {
        this.a = a;
    }
}

在这个示例中,A和B相互依赖,因为A需要B才能被创建,而B又需要A才能被创建。这种情况下,Spring IoC容器会抛出一个异常,告诉你存在循环依赖。

spring的三级缓存

什么是三级缓存,是哪三级?

  • singletonObjects:用于存放单例bean实例的缓存,即作用域为singleton的bean。当创建完一个bean后,Spring会将其放入singletonObjects缓存中。在后续获取该bean时,Spring会直接从该缓存中获取,不再需要重新创建。
  • earlySingletonObjects:用于存放创建中的bean实例的缓存。在创建一个bean的过程中,如果需要引用到该bean,则会先从earlySingletonObjects缓存中获取。如果该缓存中不存在该bean的实例,则会通过实例化工厂创建一个新的bean实例。
  • singletonFactories:用于存放创建中的bean实例的工厂缓存。在创建一个bean的过程中,如果需要引用到该bean,则会先从singletonFactories缓存中获取。如果该缓存中不存在该bean的工厂实例,则会通过实例化工厂创建一个新的bean工厂实例。在后续获取该bean时,Spring会从该缓存中获取bean工厂实例,并通过该工厂创建新的bean实例。

在以上三个缓存中,earlySingletonObjects缓存和singletonFactories缓存的作用是相同的,即存放创建中的bean实例。它们的区别在于earlySingletonObjects缓存存放的是已经创建但还没有完全初始化的bean实例,而singletonFactories缓存存放的是bean实例工厂

缓存的初始化

当我们获取一个bean时,Spring首先会从singletonObjects缓存中查找是否存在该bean的实例。如果存在,则直接返回该实例;否则,Spring会检查该bean是否在earlySingletonObjects缓存中。如果在,Spring会等待该bean完全初始化后再返回该bean实例;否则,Spring会检查singletonFactories缓存中是否存在该bean实例的工厂。如果存在,Spring会通过该工厂创建一个新的bean实例,并将其放入earlySingletonObjects缓存中,等待其完全初始化后再放入singletonObjects缓存中

缓存的清理

Spring的缓存机制采用的是弱引用的方式,因此当一个bean实例不再被引用时,会被垃圾回收机制回收。同时,Spring还提供了一些缓存清理的方法,以便在需要时手动清理缓存。具体而言,Spring提供了以下几种清理缓存的方法:

  • destroySingletons():销毁所有singleton作用域的bean实例,并清空相关的缓存。
  • clearSingletonCache():清空singletonObjects、earlySingletonObjects和singletonFactories三个缓存。
  • removeSingleton(String beanName):从缓存中移除指定bean

循环依赖的解决

spring三级缓存解决循环依赖

在 Spring 中,当出现循环依赖时,通过三级缓存解决。Spring 使用三级缓存来存储正在创建的 bean,以便在创建 bean 的过程中检测循环依赖。以下是 Spring 如何使用三级缓存解决循环依赖的步骤:

  1. 在创建 bean 的过程中,Spring 首先会将正在创建的 bean 放入三级缓存中。
  2. 当创建 bean 的过程中需要依赖另一个 bean 时,Spring 会检查三级缓存中是否存在需要依赖的 bean。如果存在,Spring 会返回缓存中的 bean 实例。
  3. 如果三级缓存中不存在需要依赖的 bean,则 Spring 会将需要依赖的 bean 标记为正在创建中,并继续创建该 bean。在创建 bean 的过程中,Spring 会将需要依赖的 bean 放入二级缓存中。
  4. 如果需要依赖的 bean 还依赖当前正在创建的 bean,则 Spring 会检测二级缓存中是否存在需要依赖的 bean。如果存在,则 Spring 会返回缓存中的 bean 实例。
  5. 如果二级缓存中不存在需要依赖的 bean,则 Spring 会将需要依赖的 bean 标记为正在创建中,并继续创建该 bean。在创建 bean 的过程中,Spring 会将需要依赖的 bean 放入一级缓存中。
  6. 在创建 bean 的过程中,如果发现依赖的 bean 依然依赖当前正在创建的 bean,Spring 将抛出 BeanCurrentlyInCreationException 异常,以避免出现死循环依赖。
  7. 当所有依赖项都已创建完毕时,Spring 会从一级缓存中获取 bean 实例,并将其放入二级缓存和三级缓存中。此时,bean 已经可以使用。

总的来说,Spring 的三级缓存是一种很好的机制,可以解决循环依赖问题,但是如果循环依赖过于复杂,还是需要手动解决。

手动解决循环依赖

当程序报错是org.springframework.beans.factory.BeanCurrentlyInCreationException,就需要手动去解决循环依赖了。在Spring中,手动解决循环依赖可以使用以下两种方式:

  1. 使用构造函数注入:将循环依赖的Bean通过构造函数注入,而不是使用属性注入。这样在创建Bean时就能够通过构造函数注入其他Bean,从而避免了循环依赖。
  2. 使用@Lazy注解:在Spring中,@Lazy注解可以将Bean的初始化时机推迟到第一次使用时。这样,如果存在循环依赖,其中一个Bean在初始化时会直接使用另一个Bean的代理对象,从而避免了循环依赖。
    ⚠️需要注意的是,手动解决循环依赖可能会导致代码复杂度提高,并且需要谨慎处理,以避免出现其他问题。因此,在实际开发中,建议尽可能地避免循环依赖,以便减少出现问题的可能性。
相关文章
|
2月前
|
缓存 架构师 Java
图解 Spring 循环依赖,一文吃透!
Spring 循环依赖如何解决,是大厂面试高频,本文详细解析,建议收藏。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
图解 Spring 循环依赖,一文吃透!
|
1月前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
57 2
|
1月前
|
缓存 NoSQL Java
Spring Boot中的分布式缓存方案
Spring Boot提供了简便的方式来集成和使用分布式缓存。通过Redis和Memcached等缓存方案,可以显著提升应用的性能和扩展性。合理配置和优化缓存策略,可以有效避免常见的缓存问题,保证系统的稳定性和高效运行。
54 3
|
1月前
|
缓存 Java 数据库连接
深入探讨:Spring与MyBatis中的连接池与缓存机制
Spring 与 MyBatis 提供了强大的连接池和缓存机制,通过合理配置和使用这些机制,可以显著提升应用的性能和可扩展性。连接池通过复用数据库连接减少了连接创建和销毁的开销,而 MyBatis 的一级缓存和二级缓存则通过缓存查询结果减少了数据库访问次数。在实际应用中,结合具体的业务需求和系统架构,优化连接池和缓存的配置,是提升系统性能的重要手段。
80 4
|
2月前
|
存储 缓存 Java
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
400 2
|
4月前
|
缓存 Java 开发工具
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
三级缓存是Spring框架里,一个经典的技术点,它很好地解决了循环依赖的问题,也是很多面试中会被问到的问题,本文从源码入手,详细剖析Spring三级缓存的来龙去脉。
238 24
|
3月前
|
缓存 Java Spring
源码解读:Spring如何解决构造器注入的循环依赖?
本文详细探讨了Spring框架中的循环依赖问题,包括构造器注入和字段注入两种情况,并重点分析了构造器注入循环依赖的解决方案。文章通过具体示例展示了循环依赖的错误信息及常见场景,提出了三种解决方法:重构代码、使用字段依赖注入以及使用`@Lazy`注解。其中,`@Lazy`注解通过延迟初始化和动态代理机制有效解决了循环依赖问题。作者建议优先使用`@Lazy`注解,并提供了详细的源码解析和调试截图,帮助读者深入理解其实现机制。
84 1
|
4月前
|
缓存 Java Spring
手写Spring Ioc 循环依赖底层源码剖析
在Spring框架中,IoC(控制反转)是一个核心特性,它通过依赖注入(DI)实现了对象间的解耦。然而,在实际开发中,循环依赖是一个常见的问题。
55 4
|
4月前
|
存储 缓存 Java
在Spring Boot中使用缓存的技术解析
通过利用Spring Boot中的缓存支持,开发者可以轻松地实现高效和可扩展的缓存策略,进而提升应用的性能和用户体验。Spring Boot的声明式缓存抽象和对多种缓存技术的支持,使得集成和使用缓存变得前所未有的简单。无论是在开发新应用还是优化现有应用,合理地使用缓存都是提高性能的有效手段。
61 1
|
8月前
|
存储 缓存 Java
【Spring原理高级进阶】有Redis为啥不用?深入剖析 Spring Cache:缓存的工作原理、缓存注解的使用方法与最佳实践
【Spring原理高级进阶】有Redis为啥不用?深入剖析 Spring Cache:缓存的工作原理、缓存注解的使用方法与最佳实践