Java面试题:Spring中的循环依赖,给程序员带来的心理阴影

java,spring · 浏览次数 : 6

小编点评

**循环依赖的解决方案** **1. 使用三级缓存机制** 三级缓存机制是一种缓存机制,在实例化一个bean时,会将该bean放入该缓存中。如果在实例化过程中需要注入其他bean,并且这个bean也正在实例化中,Spring会先从一级缓存中查找该bean的实例。如果一级缓存中没有找到,会到二级缓存中查找。如果二级缓存中也没有,那么会到三级缓存中查找ObjectFactory,并调用ObjectFactory的getObject方法来获取bean的实例。 **2. 使用 @Lazy 注解** @Lazy注解可以用于延迟bean的初始化。当一个bean被标记为@Lazy时,Spring容器在启动时不会立即实例化它,而是在第一次被使用时才进行实例化。通过将循环依赖的bean声明为懒加载,可以延迟它们的初始化过程,从而避免在容器启动时发生循环依赖问题。 **3. 避免构造器循环依赖** 构造器循环依赖是无法解决的,因为构造器在实例化bean的过程中被调用,如果两个bean相互依赖对方的构造器,那么就会形成构造器循环依赖。因此,在设计bean的依赖关系时,应该尽量避免使用构造器注入来创建循环依赖。可以使用setter注入或字段注入来代替构造器注入。 **4. 避免使用构造器注入来创建循环依赖** 在Spring中,构造器循环依赖是无法解决的,因为构造器在实例化bean的过程中被调用,如果两个bean相互依赖对方的构造器,那么就会形成构造器循环依赖。因此,在设计bean的依赖关系时,应该尽量避免使用构造器注入来创建循环依赖。可以使用setter注入或字段注入来代替构造器注入。

正文

循环依赖通常发生在两个或多个Spring Bean之间,它们通过构造器、字段(使用@Autowired)或setter方法相互依赖,从而形成一个闭环。下面是一个使用字段注入(即使用@Autowired)导致的循环依赖的示例:

 

示例代码: 

假设我们有两个类,ClassA 和 ClassB,它们相互依赖:

public class ClassA {  

    @Autowired  
    private ClassB classB;  

    // ... 其他代码 ...  
}  

@Component  
public class ClassB {  

    @Autowired  
    private ClassA classA;  

    // ... 其他代码 ...  
}

在上面的示例中,ClassA 依赖 ClassB,而 ClassB 又依赖 ClassA。当Spring容器启动时,它会尝试为这两个类创建bean的实例。但是,由于它们之间的循环依赖,这会导致问题。

问题说明: 

当Spring容器创建ClassA的bean时,它会发现ClassA依赖于ClassB,所以它会尝试创建ClassB的bean。当Spring容器创建ClassB的bean时,它又会发现ClassB依赖于ClassA,但此时ClassA的bean还没有被完全初始化(因为它正在等待ClassB的bean),这就形成了一个死循环。

 

那么Spring是如何解决这种循环依赖问题的?

 

三级缓存机制:

Spring容器在创建bean的过程中,会维护三个缓存,分别是:singletonObjects(一级缓存)、earlySingletonObjects(二级缓存)和singletonFactories(三级缓存)。

当Spring容器开始实例化一个bean时,会先将其ObjectFactory(对象工厂)放入三级缓存中。如果在实例化过程中需要注入其他bean,并且这个bean也正在实例化中,Spring会先从一级缓存中查找该bean的实例。如果没有找到,会到二级缓存中查找。如果二级缓存中也没有,那么会到三级缓存中查找ObjectFactory,并调用ObjectFactory的getObject方法来获取bean的实例。

当获取到bean的实例后,会将其放入二级缓存中,并从三级缓存中移除ObjectFactory。

当bean的实例化过程完成后,会将其放入一级缓存中。

通过这种方式,Spring可以在bean的实例化过程中解决循环依赖问题。

 

@Lazy注解:

在Spring中,可以使用@Lazy注解来延迟bean的初始化。当一个bean被标记为@Lazy时,Spring容器在启动时不会立即实例化它,而是在第一次被使用时才进行实例化。

通过将循环依赖的bean声明为懒加载,可以延迟它们的初始化过程,从而避免在容器启动时发生循环依赖问题。

需要注意的是,@Lazy注解只能用于单例作用域的bean,并且要求依赖项必须是接口类型。

 

以下是使用@Lazy注解来解决循环依赖的示例代码:

@Component  
public class ClassA {  

    private final ClassB classB;  

    // 使用@Autowired和@Lazy注解来延迟ClassB的注入  
    @Autowired  
    public ClassA(@Lazy ClassB classB) {  
        this.classB = classB;  
    }  

    // ... 其他代码 ...  
}  

@Component  
public class ClassB {  

    private final ClassA classA;  

    // 使用@Autowired和@Lazy注解来延迟ClassA的注入  
    @Autowired  
    public ClassB(@Lazy ClassA classA) {  
        this.classA = classA;  
    }  

    // ... 其他代码 ...  
}

在上面的示例中,@Lazy注解被用于ClassA和ClassB的构造器参数上,以延迟它们之间的依赖注入。这意味着在创建ClassA时,它不会立即尝试去初始化ClassB,而是会得到一个代理对象。同样,在创建ClassB时,它也不会立即初始化ClassA。

 

避免构造器循环依赖:

在Spring中,构造器循环依赖是无法解决的,因为构造器在实例化bean的过程中被调用,如果两个bean相互依赖对方的构造器,那么就会形成死锁。

因此,在设计bean的依赖关系时,应该尽量避免使用构造器注入来创建循环依赖。可以使用setter注入或字段注入来代替构造器注入。

 

下面是一个构造器循环依赖的错误代码示例:

@Component  
public class ClassA {  

    private final ClassB classB;  

    // 通过构造器注入ClassB,形成循环依赖  
    @Autowired  
    public ClassA(ClassB classB) {  
        this.classB = classB;  
    }  

    // ... 其他代码 ...  
}  

@Component  
public class ClassB {  

    private final ClassA classA;  

    // 通过构造器注入ClassA,与ClassA形成循环依赖  
    @Autowired  
    public ClassB(ClassA classA) {  
        this.classA = classA;  
    }  

    // ... 其他代码 ...  
}

在上面的示例中,ClassA和ClassB各自在构造函数中依赖于对方,这就形成了构造器循环依赖。

 

当Spring容器尝试创建这两个bean的实例时,会遇到问题: 

  • Spring首先尝试创建ClassA的实例,并发现它需要ClassB的实例。 

  • 然后,Spring尝试创建ClassB的实例,但发现它需要ClassA的实例。 

  • 由于ClassA的实例正在等待ClassB的实例,而ClassB的实例又正在等待ClassA的实例,这导致了一个死循环,无法继续创建bean的实例。

 

综上所述,Spring通过三级缓存机制、@Lazy注解以及避免构造器循环依赖等方式来解决循环依赖问题。这些机制使得Spring容器能够更加灵活地处理bean之间的依赖关系,提高系统的可维护性和可扩展性。

 

与Java面试题:Spring中的循环依赖,给程序员带来的心理阴影相似的内容:

Java面试题:Spring中的循环依赖,给程序员带来的心理阴影

循环依赖通常发生在两个或多个Spring Bean之间,它们通过构造器、字段(使用@Autowired)或setter方法相互依赖,从而形成一个闭环。Spring通过三级缓存机制、@Lazy注解以及避免构造器循环依赖等方式来解决循环依赖问题。这些机制使得Spring容器能够更加灵活地处理bean之间...

Java面试题:Spring框架除了IOC和AOP,还有哪些好玩的设计模式?

Spring是一个基于Java的企业级应用程序开发框架,它使用了多种设计模式来实现其各种特性和功能。本文将介绍一些在Spring中使用的常见设计模式以及相应的代码示例和说明。

Java面试题:你知道Spring的IOC吗?那么,它为什么这么重要呢?

Spring的IOC(控制反转)是一种设计模式,它允许开发者将对象的创建和管理交给Spring框架来完成。在Spring中,IOC允许开发者将对象依赖关系从代码中分离出来,从而使代码更加灵活、可重用和易于管理。 IoC 全称Inverse of Control(反向控制或控制反转)。 在类和类之间存

Java面试题:Spring Bean线程安全?别担心,只要你不写并发代码就好了!

Spring Bean是单例模式,即在整个应用程序上下文中只有一个实例。在多线程环境下,Singleton Scope Bean可能会发生线程安全问题。Spring Bean是否线程安全取决于Bean的作用域和Bean本身的实现。在使用Singleton Scope Bean时需要特别注意线程安全问...

Java面试题:@PostConstruct、init-method和afterPropertiesSet执行顺序?

在Spring框架中,@PostConstruct注解、init-method属性、以及afterPropertiesSet()方法通常用于初始化Bean的逻辑。它们都提供了在Bean创建和初始化完成后执行的方法,但执行顺序有所不同。

Java面试题:SpringBoot异常捕获,让程序“免疫”一切错误!

在Spring Boot应用程序中,捕获全局异常是一个重要的方面,它可以帮助我们处理在应用程序运行时可能发生的各种错误情况。通过适当地捕获和处理这些异常,我们可以改善用户体验并及时采取必要的措施。

京东面试:SpringBoot同时可以处理多少请求?

Spring Boot 作为 Java 开发中必备的框架,它为开发者提供了高效且易用的开发工具,所以和它相关的面试题自然也很重要,咱们今天就来看这道经典的面试题:SpringBoot同时可以处理多少个请求 ? 准确的来说,Spring Boot 同时可以处理多少个请求,并不取决于 Spring Bo

聊聊Spring Cloud Alibaba解决方案组件

在java的微服务解决方案中,最先出现目前应用比较多的就是spring cloud netfix系列,但是随着阿里的强劲支持,spring cloud alibaba解决方案逐渐可以替代前者,当然dubbo也是不容小觑的。之前面试几家公司应用的都是spring cloud alibaba,随着我自己

阿里也出手了!Spring CloudAlibaba AI问世了

写在前面 在之前的文章中我们有介绍过SpringAI这个项目。SpringAI 是Spring 官方社区项目,旨在简化 Java AI 应用程序开发, 让 Java 开发者想使用 Spring 开发普通应用一样开发 AI 应用。 而SpringAI 主要面向的是国外的各种大模型接入,对于国内开发者可

Java面试题:让依赖注入变得简单,面对@Autowired和@Resource,该如何选择?

@Autowired是Spring框架提供的注解,@Resource是Java EE 5规范提供的注解。 @Autowired默认按照类型自动装配,而@Resource默认按照名称自动装配。 @Autowired支持@Qualifier注解来指定装配哪一个具有相同类型的bean,而@Resourc...