当前位置: 首页 > wzjs >正文

建设网站费用明细手表商城网站建设

建设网站费用明细,手表商城网站建设,网站后台公告id修改,交互设计留学🚨 Spring Boot循环依赖全解析:原理、解决方案与最佳实践 #SpringBoot核心 #依赖注入 #设计模式 #性能优化 一、循环依赖的本质与危害 1.1 什么是循环依赖? 循环依赖指两个或多个Bean相互直接或间接引用,形成闭环依赖关系。 典…

🚨 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 'serviceA':  
Requested bean is currently in creation: Is there an unresolvable circular reference?  

1.2 核心危害

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

二、Spring的三级缓存机制

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

2.1 三级缓存结构

缓存级别存储内容
一级缓存(singletonObjects)完全初始化的Bean
二级缓存(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 {  private 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.springframework.boot</groupId>  <artifactId>spring-boot-maven-plugin</artifactId>  <configuration>  <excludes>  <exclude>  <groupId>org.projectlombok</groupId>  <artifactId>lombok</artifactId>  </exclude>  </excludes>  </configuration>  
</plugin>  

运行命令:

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

5.3 架构规范

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

六、常见问题解答

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

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

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

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

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

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

七、总结与最佳实践

黄金法则

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

紧急修复流程

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

工具推荐

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

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

http://www.dtcms.com/wzjs/807044.html

相关文章:

  • 旅游网站排行榜前十名官网上海网站建设品
  • 温州建校网站wordpress开发的主流架构
  • 网上做家教那个网站好站长工具在线查询
  • 网页制作与网站建设...东道设计公司规模
  • 在线做简历的网站微信运营是干嘛的
  • 网站架构图文章生成器网页版
  • 设计网站策划书项目商业网站建设方案
  • 接单做网页的网站网站推广的方法和途径
  • 深圳手机模板建站网站建设的步骤图片过程
  • 网站建设的功能特点有哪些龙岩网站制作教程
  • 索牛网站建设5ucms和wordpress的区别
  • 网站翻页模板最强的网站建设电话
  • 怎样做类似淘宝的网站网上有做衣服的网站有哪些
  • 文登区做网站的公司asp.net网站开发与项目实战
  • 怎样申请一个免费的网站空间朋友做的网站图片不显示不出来
  • 未来做哪些网站致富一个完整的产品运营方案
  • 网站重建看谁做的好舞蹈视频网站
  • 我的个人网页设计效果图镇江网站排名优化费用
  • 网站制作河南网页的后缀名有那些
  • 浙江省网站建设与管理试卷wordpress站内搜索框
  • 通州富阳网站建设wordpress 文件大小
  • 建设网站报价表代理平台推荐
  • 织梦网站管理安装手机wap网站特效
  • 光谷网站建设哪家好模板页
  • 单本小说网站浙江大学微纳加工平台
  • 海淀网站建设公司排名轻创优选地推app
  • 江西省建设协会网站php做调查问卷网站
  • 美容美发网站源码把网站从空间删除
  • 研发网站要多长时间南通网站建设公司哪家好
  • 厦门网站推广步骤机构wordpress插件语言包