【Java面试】Spring如何解决循依赖问题?高手的回答你能给多少分?

“Spring如何解决循环依赖问题”!

这是一道非常经典并且考察频率很高的面试题。

大家好,我是Mic,一个工作了14年的Java程序员

下面我们来分析一下面试官对于这个问题的考察意图

考察目的

这个问题考察的求职者年限还挺广的,工作1年到工作7年之间都会遇到。

我认为考察目的有三个:

  • Spring 是Java开发的基础应用框架,所以考察的是基本功

  • 考察技术深度,判断你的能力高低以及

  • 作为人才筛选的区分度

因此对于这个问题,不仅仅要回答原因,还需要有自己的思考和总结。

另外还需要注意,这个问题问的是Spring如何解决循环依赖,因此对于Spring无法解决的循环依赖我们没必要在这里展开。

下面我们来分析一下这个问题的产生背景和解决方案。

问题解析

循环依赖是指一个或多个 Bean 实例之间存在直接或间接的依赖关系,构成循环调用。

通常表现为三种形态。

  • 互相依赖,也就是A依赖B,B依赖A

  • 间接依赖,两个以上的Bean存在间接依赖关系造成循环调用。

  • 自我依赖,自己依赖自己造成了循环依赖

Spring本身也考虑到了这方面的问题,所以它设计了三级缓存来解决部分循环依赖的问题。

所谓三级缓存,其实就是用来存放不同类型的Bean。

  • 第一级缓存存放完全初始化好的Bean,这个Bean可以直接使用了

  • 第二级缓存存放原始的Bean对象,也就是说Bean里面的属性还没有进行赋值

  • 第三级缓存存放Bean工厂对象,用来生成原始Bean对象并放入到二级缓存中

假设BeanA和BeanB存在循环依赖,那么在三级缓存的设计下,我画了这样一个图来描述工作原理。

  • 初始化BeanA,先把BeanA实例化,然后把BeanA包装成ObjectFactory对象保存到三级缓存中。

  • 接着BeanA开始对属性BeanB进行依赖注入,于是开始初始化BeanB,同样做两件事,

    创建BeanB实例,以及加入到三级缓存。

  • 然后,BeanB也开始进行依赖注入,在三级缓存中找到了BeanA,于是完成BeanA的依赖注入

  • BeanB初始化成功以后保存到一级缓存,于是BeanA可以成功拿到BeanB的实例,从而完成正常的依赖注入。

整个流程看起来很复杂,但是它的核心思想就是把Bean的实例化和Bean中属性的依赖注入这两个过程分离出来。

不过要注意的是,Spring本身只能解决单实例存在的循环引用问题,但是存在以下四种情况需要人为干预:

  • 多实例的Setter注入导致的循环依赖,需要把Bean改成单例。

  • 构造器注入导致的循环依赖,可以通过@Lazy注解

  • DependsOn导致的循环依赖,找到注解循环依赖的地方,迫使它不循环依赖。

  • 单例的代理对象Setter注入导致的循环依赖

    • 可以使用@Lazy注解。

    • 或者使用@DependsOn注解指定加载先后关系。

在实际开发中,出现循环依赖的根本原因还是在代码设计的时候,因为模块的耦合度较高,依赖关系复杂导致的,我们应该尽可能的从系统设计角度去考虑模块之间的依赖关系,避免循环依赖的问题。

高手:

Spring设计了三级缓存来解决循环依赖问题。

  • 第一级缓存里面存储完整的Bean实例,这些实例是可以直接被使用的。

  • 第二级缓存里面存储的是实例化以后,但是还没有设置属性值的Bean实例,也就是Bean里面的依赖注入还没有做。

  • 第三级缓存用来存放Bean工厂,它主要用来生成原始Bean对象并且放到第二级缓存里面。

三级缓存的核心思想,就是把Bean的实例化,和Bean里面的依赖注入进行分离。

采用一级缓存存储完整的Bean实例,采用二级缓存来存储不完整的Bean实例,通过不完整的Bean实例作为突破口,解决循环依赖的问题。

至于第三级缓存,主要是解决代理对象的循环依赖问题。

以上就是我的理解。

总结

下次面试的时候遇到这个问题,大家知道怎么回答了吗?

如果你喜欢我的作品,记得点赞收藏加关注哦!!!

另外,我将所有Java面试系列制作成了完整的面试文档。它的便捷之处在于,可以通过检索的方式,找到你想要的面试题,目前已经更新200期,总计超过20W字!

【想领取面试文档的小伙伴可以点击文章底部名片无套路免费赠送给大家!】

需要高手面试文档面试文档的小伙伴可以扫描下方二维码
↓↓↓↓↓↓↓↓↓↓↓↓↓

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是一些常见的 Spring 面试题以及回答建议: 1. 什么是 SpringSpring 的优点有哪些? 回答建议:Spring 是一个轻量级的开源框架,用于构建企业级应用程序。它提供了一种依赖注入和面向切面编程的方式,使得应用程序开发更简单和更容易测试,并且提供了很多现成的模块,如数据访问、Web开发等。Spring 的优点包括:简化应用程序开发、提高应用程序的可测试性、提高应用程序的可扩展性和可维护性、提高代码的重用性等等。 2. Spring 的核心模块是哪些?它们各自的功能是什么? 回答建议:Spring 的核心模块包括:Spring Core、Spring Context、Spring AOP、Spring DAO、Spring ORM、Spring Web、Spring MVC、Spring Web Services 等。Spring Core 模块是 Spring 框架的核心,提供了 Spring 框架的基本功能,如依赖注入、控制反转等。Spring Context 模块是 Spring 框架的上下文,提供了 Spring 应用程序中的所有对象的访问点。Spring AOP 模块提供了面向切面编程的支持。Spring DAO 模块为数据访问提供了支持。Spring ORM 模块为对象关系映射提供了支持。Spring Web 模块为 Web 应用程序开发提供了支持。Spring MVC 模块是 Spring 框架的 MVC 框架。Spring Web Services 模块为 Web 服务开发提供了支持。 3. Spring 中 Bean 是什么?Bean 的作用域有哪些? 回答建议:在 Spring 中,Bean 是一个由 Spring IOC 容器管理的对象。Bean 可以是任何一个 Java 对象,如 POJO、Service、DAO、Entity 等。Bean 的作用域有五种:Singleton、Prototype、Request、Session 和 Global Session。Singleton 作用域表示在整个应用程序中只有一个 Bean 实例。Prototype 作用域表示每次从容器中获取 Bean 时都会创建一个新的实例。Request 作用域表示在一次 HTTP 请求中只有一个 Bean 实例。Session 作用域表示在一个 HTTP Session 中只有一个 Bean 实例。Global Session 作用域表示在一个全局的 HTTP Session 中只有一个 Bean 实例。 4. Spring 中的依赖注入是什么?有哪些方式可以实现依赖注入? 回答建议:依赖注入是 Spring 的一个核心特性,它允许我们将对象之间的依赖关系从代码中分离出来,交给 Spring IOC 容器处理。Spring 中有三种方式实现依赖注入:构造方法注入、Setter 方法注入和接口注入。构造方法注入是通过构造方法来注入依赖项。Setter 方法注入是通过 Setter 方法来注入依赖项。接口注入是通过实现一个特定的接口来注入依赖项。 5. Spring AOP 是什么?它的作用是什么?它的实现方式有哪些? 回答建议:Spring AOP 是 Spring 框架中的一个模块,它提供了面向切面编程的支持。AOP 的作用是将应用程序的横切关注点(如日志、事务、安全等)从应用程序主体中分离出来,并将它们封装成可重用的模块。Spring AOP 的实现方式有两种:使用 JDK 动态代理和使用 CGLIB(Code Generation Library)动态代理。 6. Spring 中的事务管理是如何实现的? 回答建议:Spring 中的事务管理是通过 Spring 的事务管理器来实现的。Spring 的事务管理器是一个用于管理事务的对象,它提供了一系列的方法来控制事务的提交和回滚。Spring 支持编程式事务管理和声明式事务管理。编程式事务管理是通过编写代码来控制事务的提交和回滚。声明式事务管理是通过配置文件来控制事务的提交和回滚。 7. Spring MVC 的工作原理是什么?它的核心组件有哪些? 回答建议:Spring MVC 是一个基于 MVC(Model-View-Controller)架构的 Web 框架,它提供了一种简单和易于使用的方式来创建 Web 应用程序。Spring MVC 的工作原理是:当用户发送请求时,请求将被控制器接收,控制器将调用模型来处理请求,并将处理结果返回给视图。Spring MVC 的核心组件包括:DispatcherServlet、HandlerMapping、Controller、ViewResolver 等。 8. Spring Boot 是什么?它的优点有哪些?如何使用 Spring Boot 创建一个 Web 应用程序? 回答建议:Spring Boot 是一个用于构建现代化 Web 应用程序的框架,它提供了一种简单和易于使用的方式来创建、配置和部署应用程序。Spring Boot 的优点包括:快速构建应用程序、自动配置应用程序、提供现成的模块、易于部署和维护等。使用 Spring Boot 创建一个 Web 应用程序的步骤如下:创建一个 Spring Boot 项目、添加 Web 依赖项、创建一个控制器、创建一个视图、运行应用程序。 9. Spring Cloud 是什么?它的主要组件有哪些? 回答建议:Spring Cloud 是一个用于构建分布式系统的框架,它提供了一系列的组件来帮助开发人员构建和部署分布式应用程序。Spring Cloud 的主要组件包括:服务注册与发现、负载均衡、断路器、配置中心、消息总线等。 10. Spring Security 是什么?它的作用是什么?它的常见使用场景有哪些? 回答建议:Spring Security 是一个用于保护 Web 应用程序安全的框架,它提供了一种简单和易于使用的方式来实现身份验证、授权和其他安全功能。Spring Security 的作用是保护 Web 应用程序免受各种安全攻击,如跨站点脚本、SQL 注入、会话劫持等。Spring Security 的常见使用场景包括:基于角色的访问控制、基于资源的访问控制、单点登录等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

跟着Mic学架构

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值