Spring Boot循环依赖原理、解决方案与最佳实践(全解析)

本文主要是介绍Spring Boot循环依赖原理、解决方案与最佳实践(全解析),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《SpringBoot循环依赖原理、解决方案与最佳实践(全解析)》循环依赖指两个或多个Bean相互直接或间接引用,形成闭环依赖关系,:本文主要介绍SpringBoot循环依赖原理、解决方案与最...

Spring Boot循环依赖全解析:原理、解决方案与最佳实践

#SpringBoot核心 #依赖注入 #设计模式 #性能优化

一、循环依赖的本质与危害

1.1 什么是循环依赖?

循环依赖指两个或多个Bean相互直接或间接引用,形成闭环依赖关系。
典型场景

@Service  
public class ServiceA {  
    @Autowired  
    private ServiceB serviceB;  
}  
@Service  
public class ServiceB {  
    @Autowired  
    private ServiceA serviceA;  
}  

Spring启动时抛出异常:

BeanCurrentlyInCreationException: Error creating bean with name 'sJwBEyIerviceA':  
Requested bean is currently in creation: Is there an unresolvable circular reference?  

1.2 核心危害

  • 应用启动失败:Spring无法完成Bean初始化
  • 设计缺陷信号:模块职责不清,耦合度过高
  • 潜在性能问题:即使解决循环依赖,可能引发隐藏的初始化顺序问题

pring无法完成Bean初始化设计缺陷信号:模块职责不清,耦合度过高潜在性能问题:即使解决循环依赖,可能引发隐藏的初始化顺序问题

二、Spring的三级缓存机制

Spring通过三级缓存解决Setter/Field注入的循环依赖,但无法解决构造器注入的循环依赖。

2.1 三级缓存结构

缓存级别存储内容
一级缓存(singletonObjects)完全初始化的Bean
二级缓存php(earlySingletonObjects)提前暴露的早期Bean(未完成属性填充)
三级缓存(singletonFactories)Bean工厂对象(用于生成早期引用)

 2.2 解决流程(以ServiceA和ServiceB为例)

1. 创建ServiceA → 将原始对象工厂放入三级缓存  
2. 填充ServiceA属性 → 发现需要ServiceB  
3. 创建ServiceB → 将原始对象工厂放入三级缓存  
4. 填充ServiceB属性 → 从三级缓存获取ServiceA的工厂 → 生成代理对象  
5. ServiceB初始化完成 → 移入一级缓存  
6. ServiceA继续填充ServiceB → 从一级缓存获取ServiceB → 完成初始化  

三、解决方案与代码实战

3.1 避免构造器注入循环

构造器注入循环依赖无法解决(Spring 5.3+默认禁止):

// 错误示例:启动直接失败  
@Service  
public class ServiceA {  
    private final ServiceB serviceB;  
    public ServiceA(ServiceB serviceB) {  
        this.serviceB = serviceB;  
    }  
}  
@Service  
public class ServiceB {  
    private final ServiceA serviceA;  
    public ServiceB(ServiceA serviceA) {  
        this.serviceA = serviceA;  
    }  
}  

强制允许构造器循环依赖(不推荐)

# application.properties  
spring.main.allow-circular-references=true  

3.2 使用Setter/Field注入

将构造器注入改为Setter注入:

@Service  
public class ServiceA {  
    private ServiceB serviceB;  
    @Autowired  
    public void setServiceB(ServiceB serviceB) {  
        this.serviceB = serviceB;  
    }  
}  
@Service  
public class ServiceB {  
    private ServiceA serviceA;  
    @Autowired  
    public void setServiceA(ServiceA serviceA) {  
        this.serviceA = serviceA;  
    }  
}  

3.3 @Lazy延迟加载

强制延迟其中一个Bean的初始化:

@Service  
public class ServiceA {  
    @Lazy  
    @Autowired  
    private ServiceB serviceB;  
}  

原理:ServiceB被代理,首次调用时才会真实初始化。

3.4 接口抽象解耦

通过接口隔离实现类依赖

public interface IServiceA {  
    void DOSomething();  
}  
public interface IServiceB {  
    void doAnother();  
}  
@Service  
public class ServiceAImpl implements IServiceA {  
    @Autowired  
    private IServiceB serviceB;  
}  
@Service  
public class ServiceBImpl implements IServiceB {  
    @Autowired  
    private IServiceA serviceA;  
}  

四、深度优化:设计模式应用

4.1 依赖倒置原则(DIP)

高层模块不应依赖低层模块,二者都应依赖抽象

// 定义数据访问接口  
public interface UserRepository {  
    User findById(Long id);  
}  
// 高层服务依赖接口  
@Service  
public class UserService {  
    prjavascriptivate final UserRepository repository;  
    public UserService(UserRepository repository) {  
        this.repository = repository;  
    }  
}  
// 低层实现  
@Repository  
public class JpaUserRepository implements UserRepository {  
    // 实现细节  
}  

4.2 事件驱动模型

通过ApplicationEvent解耦强依赖

// ServiceA发布事件  
@Service  
public class ServiceA {  
    @Autowired  
    private ApplicationEventPublisher publisher;  
    public void triggerEvent() {  
        publisher.publishEvent(new CustomEvent(this));  
    }  
}  
// ServiceB监听事件  
@Service  
public class ServiceB {  
    @EventListener  
    public void handleEvent(CustomEvent event) {  
        // 处理逻辑  
    }  
}  

五、检测与预防工具

5.1 IDE检测

  • IntelliJ IDEA:自动标记循环依赖(需安装Spring Assistant插件)
  • Eclipse:通过STS (Spring Tool Suite)插件提示

5.2 Maven插件分析

<plugin>  
    <groupId>org.spriwww.chinasem.cnngframework.boot</groupId>  
    <artifactId>spring-boot-maven-plugin</artifactId>  
    <configuration>  
        <excludes>  
            <exclude>  
                <groupId>org.projectlombok</groupId>  
                <artifactId>lombok</artifactId>  
            </exclude>  
        </excludes&gpythont;  
    </configuration>  
</plugin>  

运行命令:

mvn spring-boot:run -Dspring-boot.run.profiles=dev  

5.3 架构规范

  • 模块化设计:按业务拆分模块(如user-serviceorder-service
  • 依赖规则
    • 下层模块可依赖上层
    • 同层禁止相互依赖
    • 通用工具类下沉至common模块

六、常见问题解答

Q1:允许循环依赖对性能有影响吗?

  • 短期影响:增加Bean创建时的上下文切换
  • 长期风险:可能导致内存泄漏(如未正确释放代理对象)

Q2:@Lazy注解可以随便用吗?

  • 慎用场景:频繁调用的Bean会增加代理开销
  • 最佳实践:仅用于解决无法重构的历史代码

Q3:Spring为什么能解决Setter注入循环依赖?

  • 核心机制:三级缓存提前暴露未完成初始化的对象引用

七、总结与最佳实践

黄金法则

  • 优先使用构造器注入:强制暴露依赖关系
  • 遵守单一职责原则:拆分超过500行代码的类
  • 定期依赖审查:使用ArchUnit等工具检测架构规范

紧急修复流程

发现循环依赖 → 使用@Lazy临时解决 → 标记为技术债务 → 排期重构  

工具推荐

  • ArchUnit:架构规则检测
  • Spring Boot Actuator:运行时依赖分析

通过合理设计+规范约束,可有效避免循环依赖,构建高可维护的Spring Boot应用!

到此这篇关于Spring Boot循环依赖全解析:原理、解决方案与最佳实践的文章就介绍到这了,更多相关Spring Boot循环依赖内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程China编程(www.chinasem.cn)!

这篇关于Spring Boot循环依赖原理、解决方案与最佳实践(全解析)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1154270

相关文章

C#中async await异步关键字用法和异步的底层原理全解析

《C#中asyncawait异步关键字用法和异步的底层原理全解析》:本文主要介绍C#中asyncawait异步关键字用法和异步的底层原理全解析,本文给大家介绍的非常详细,对大家的学习或工作具有一... 目录C#异步编程一、异步编程基础二、异步方法的工作原理三、代码示例四、编译后的底层实现五、总结C#异步编程

在Spring Boot中浅尝内存泄漏的实战记录

《在SpringBoot中浅尝内存泄漏的实战记录》本文给大家分享在SpringBoot中浅尝内存泄漏的实战记录,结合实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录使用静态集合持有对象引用,阻止GC回收关键点:可执行代码:验证:1,运行程序(启动时添加JVM参数限制堆大小):2,访问 htt

SpringBoot集成Milvus实现数据增删改查功能

《SpringBoot集成Milvus实现数据增删改查功能》milvus支持的语言比较多,支持python,Java,Go,node等开发语言,本文主要介绍如何使用Java语言,采用springboo... 目录1、Milvus基本概念2、添加maven依赖3、配置yml文件4、创建MilvusClient

浅析Java中如何优雅地处理null值

《浅析Java中如何优雅地处理null值》这篇文章主要为大家详细介绍了如何结合Lambda表达式和Optional,让Java更优雅地处理null值,感兴趣的小伙伴可以跟随小编一起学习一下... 目录场景 1:不为 null 则执行场景 2:不为 null 则返回,为 null 则返回特定值或抛出异常场景

Python如何自动生成环境依赖包requirements

《Python如何自动生成环境依赖包requirements》:本文主要介绍Python如何自动生成环境依赖包requirements问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录生成当前 python 环境 安装的所有依赖包1、命令2、常见问题只生成当前 项目 的所有依赖包1、

Node.js 数据库 CRUD 项目示例详解(完美解决方案)

《Node.js数据库CRUD项目示例详解(完美解决方案)》:本文主要介绍Node.js数据库CRUD项目示例详解(完美解决方案),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考... 目录项目结构1. 初始化项目2. 配置数据库连接 (config/db.js)3. 创建模型 (models/

SpringMVC获取请求参数的方法

《SpringMVC获取请求参数的方法》:本文主要介绍SpringMVC获取请求参数的方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下... 目录1、通过ServletAPI获取2、通过控制器方法的形参获取请求参数3、@RequestParam4、@

SpringBoot应用中出现的Full GC问题的场景与解决

《SpringBoot应用中出现的FullGC问题的场景与解决》这篇文章主要为大家详细介绍了SpringBoot应用中出现的FullGC问题的场景与解决方法,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录Full GC的原理与触发条件原理触发条件对Spring Boot应用的影响示例代码优化建议结论F

springboot项目中常用的工具类和api详解

《springboot项目中常用的工具类和api详解》在SpringBoot项目中,开发者通常会依赖一些工具类和API来简化开发、提高效率,以下是一些常用的工具类及其典型应用场景,涵盖Spring原生... 目录1. Spring Framework 自带工具类(1) StringUtils(2) Coll

Python 中的 with open文件操作的最佳实践

《Python中的withopen文件操作的最佳实践》在Python中,withopen()提供了一个简洁而安全的方式来处理文件操作,它不仅能确保文件在操作完成后自动关闭,还能处理文件操作中的异... 目录什么是 with open()?为什么使用 with open()?使用 with open() 进行