Spring Boot 启动时循环依赖的详细排查和解决步骤

Spring Boot 启动时循环依赖的详细排查和解决步骤

在 Spring Boot 应用中,循环依赖指的是两个或多个 Bean 互相依赖,形成一个闭环。这会导致 Spring 容器无法正常启动。以下是详细的排查和解决步骤:

1. 排查循环依赖

查看日志
  1. 启动日志 :首先,查看启动日志中是否有关于循环依赖的异常信息。通常,日志会指出哪个 Bean 存在循环依赖问题。
    • 示例日志

      复制代码
      org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'beanA': Requested bean is currently in creation: Is there an unresolvable circular reference?
分析 Bean 定义
  1. 使用 IDE 工具

    • IntelliJ IDEA :可以使用 Spring 工具窗口来查看 Bean 的依赖关系。
    • Eclipse :可以使用 Spring Tool Suite 插件来查看 Bean 依赖图。
  2. 手动检查

    • 查看所有 @Component@Service@Repository@Configuration 注解的类,检查它们的依赖关系。
检查注入方式
  1. 构造函数注入

    • 如果使用构造函数注入,确保没有形成循环依赖。例如:

      java 复制代码
      @Component
      public class A {
          private final B b;
      
          @Autowired
          public A(B b) {
              this.b = b;
          }
      }
      
      @Component
      public class B {
          private final A a;
      
          @Autowired
          public B(A a) {
              this.a = a;
          }
      }
  2. Setter 注入

    • 如果使用 setter 注入,请确认是否有循环依赖。例如:

      java 复制代码
      @Component
      public class A {
          private B b;
      
          @Autowired
          public void setB(B b) {
              this.b = b;
          }
      }
      
      @Component
      public class B {
          private A a;
      
          @Autowired
          public void setA(A a) {
              this.a = a;
          }
      }

2. 解决循环依赖

使用 @Lazy 注解
  1. 构造函数注入的循环依赖
    • 使用 @Lazy 注解来延迟 Bean 的初始化。示例如下:

      java 复制代码
      @Component
      public class A {
          private final B b;
      
          @Autowired
          public A(@Lazy B b) {
              this.b = b;
          }
      }
      
      @Component
      public class B {
          private final A a;
      
          @Autowired
          public B(@Lazy A a) {
              this.a = a;
          }
      }
使用 Setter 注入
  1. 将构造函数注入改为 Setter 注入
    • Setter 注入可以在 Bean 创建后再设置依赖,从而打破循环依赖。例如:

      java 复制代码
      @Component
      public class A {
          private B b;
      
          @Autowired
          public void setB(B b) {
              this.b = b;
          }
      }
      
      @Component
      public class B {
          private A a;
      
          @Autowired
          public void setA(A a) {
              this.a = a;
          }
      }
重构 Bean 设计
  1. 拆分 Bean
    • 重新设计 Bean 的依赖关系,将依赖关系分开,避免直接的循环依赖。例如:

      java 复制代码
      @Component
      public class A {
          private final C c;
      
          @Autowired
          public A(C c) {
              this.c = c;
          }
      }
      
      @Component
      public class B {
          private final C c;
      
          @Autowired
          public B(C c) {
              this.c = c;
          }
      }
      
      @Component
      public class C {
          private final A a;
          private final B b;
      
          @Autowired
          public C(A a, B b) {
              this.a = a;
              this.b = b;
          }
      }
使用 @Configuration
  1. @Configuration 类中定义 Bean
    • @Configuration 类中手动创建 Bean,可以更灵活地控制 Bean 的创建顺序。例如:

      java 复制代码
      @Configuration
      public class AppConfig {
      
          @Bean
          public A a(B b) {
              return new A(b);
          }
      
          @Bean
          public B b(A a) {
              return new B(a);
          }
      }

示例代码

假设你有两个类 AB,它们之间存在循环依赖。以下是如何使用 @Lazy 注解来解决问题的示例:

java 复制代码
@Component
public class A {
    private final B b;

    @Autowired
    public A(@Lazy B b) {
        this.b = b;
    }
}

@Component
public class B {
    private final A a;

    @Autowired
    public B(@Lazy A a) {
        this.a = a;
    }
}

总结

  • 分析和排查:查看日志、使用 IDE 工具、检查 Bean 定义和依赖注入方式。
  • 解决策略 :使用 @Lazy 注解、改用 Setter 注入、重构 Bean 设计、在 @Configuration 类中定义 Bean。

通过这些步骤,你可以有效地解决 Spring Boot 应用中的循环依赖问题,确保应用能够顺利启动。

一般使用 @Lazy 注解就可以解决。

仅为个人知识分享及开发中遇到的问题总结,

希望对你有所帮助,若有问题欢迎指正~😊

相关推荐
abcnull5 小时前
用javaparser做精准测试
java·ast·静态代码分析·精准测试·javaparser
叶小鸡5 小时前
Java 篇-项目实战-苍穹外卖-笔记汇总
java·开发语言·笔记
Johnstons5 小时前
Wireshark ExpertInfo是什么?一文讲透异常分级、适用场景、和传统抓包阅读的区别与排查标准
网络·测试工具·wireshark·es
AI人工智能+电脑小能手5 小时前
【大白话说Java面试题】【Java基础篇】第22题:HashMap 和 HashSet 有哪些区别
java·开发语言·哈希算法·散列表·hash
alxraves5 小时前
医疗器械软件注册指导原则注意事项
网络·安全·健康医疗·制造
juniperhan5 小时前
Flink 系列第21篇:Flink SQL 函数与 UDF 全解读:类型推导、开发要点与 Module 扩展
java·大数据·数据仓库·分布式·sql·flink
ID_180079054735 小时前
Python 实现亚马逊商品详情 API 数据准确性校验(极简可用 + JSON 参考)
java·python·json
devpotato6 小时前
Spring Boot mTLS 报 `keystore password was incorrect`:不一定是密码错了
spring boot·tls·pkcs12·mtls
c++之路6 小时前
C++23概述
java·c++·c++23
专注API从业者7 小时前
Open Claw 京东商品监控选品实战:一键抓取、实时监控、高效选品
java·服务器·数据库