当前位置: 代码网 > it编程>编程语言>Java > springboot循环依赖问题案例代码及解决办法

springboot循环依赖问题案例代码及解决办法

2025年04月03日 Java 我要评论
在 spring boot 中,如果两个或多个 bean 之间存在循环依赖(即 bean a 依赖 bean b,而 bean b 又依赖 bean a),会导致 spring 的依赖注入机制无法正确

在 spring boot 中,如果两个或多个 bean 之间存在循环依赖(即 bean a 依赖 bean b,而 bean b 又依赖 bean a),会导致 spring 的依赖注入机制无法正确处理,从而抛出异常。以下是循环依赖的场景分析、案例代码及常见的解决方法。

1. 什么是循环依赖?

概念

循环依赖是指两个或多个 bean 相互依赖,形成一个闭环。例如:

  • 直接循环依赖
    • a 依赖 bb 依赖 a
  • 间接循环依赖
    • a 依赖 bb 依赖 c,而 c 又依赖 a

2. 循环依赖的场景案例

以下是一个简单的直接循环依赖场景。

案例代码 bean a

@component
public class beana {
    @autowired
    private beanb beanb;
    public beana() {
        system.out.println("beana constructor");
    }
    public void dosomething() {
        system.out.println("beana is doing something");
    }
}

bean b

@component
public class beanb {
    @autowired
    private beana beana;
    public beanb() {
        system.out.println("beanb constructor");
    }
    public void dosomething() {
        system.out.println("beanb is doing something");
    }
}

启动类

@springbootapplication
public class circulardependencydemoapplication {
    public static void main(string[] args) {
        springapplication.run(circulardependencydemoapplication.class, args);
    }
}

运行结果

运行时,spring 会抛出以下异常:

caused by: org.springframework.beans.factory.beancurrentlyincreationexception:
error creating bean with name 'beana':
requested bean is currently in creation: is there an unresolvable circular reference?

3. 解决循环依赖的常见方法

方法 1:使用 @lazy 注解

@lazy 注解可以延迟加载 bean,使 spring 在真正需要使用时才注入依赖,从而打破循环依赖。

修改代码

在其中一个依赖上添加 @lazy 注解:

beana:

@component
public class beana {
    @autowired
    @lazy
    private beanb beanb;
    public beana() {
        system.out.println("beana constructor");
    }
    public void dosomething() {
        system.out.println("beana is doing something");
    }
}

beanb:

@component
public class beanb {
    @autowired
    private beana beana;
    public beanb() {
        system.out.println("beanb constructor");
    }
    public void dosomething() {
        system.out.println("beanb is doing something");
    }
}

运行结果

程序运行正常,输出:

beana constructor
beanb constructor

方法 2:使用构造器注入解决循环依赖

spring 无法通过构造器注入解决循环依赖,因为在构造器注入的过程中,所有依赖必须在实例化时完全注入。这种情况下,需要重新设计代码结构以打破循环依赖。

重构代码

将循环依赖重构为单向依赖。例如,可以引入第三方协作者 bean 来解耦。

beana:

@component
public class beana {
    private final helper helper;
    public beana(helper helper) {
        this.helper = helper;
        system.out.println("beana constructor");
    }
    public void dosomething() {
        system.out.println("beana is doing something");
    }
}

beanb:

@component
public class beanb {
    private final helper helper;
    public beanb(helper helper) {
        this.helper = helper;
        system.out.println("beanb constructor");
    }
    public void dosomething() {
        system.out.println("beanb is doing something");
    }
}

helper:

@component
public class helper {
    public void assist() {
        system.out.println("helper is assisting");
    }
}

运行结果

helper constructor
beana constructor
beanb constructor

通过引入 helperbeanabeanb 不再直接依赖彼此,循环依赖被消除。

方法 3:使用 @postconstruct 或 setter 注入

通过构造器注入会导致循环依赖问题,但可以使用 setter 方法注入来延迟依赖注入的时机。

修改代码

beana:

@component
public class beana {
    private beanb beanb;
    public beana() {
        system.out.println("beana constructor");
    }
    @autowired
    public void setbeanb(beanb beanb) {
        this.beanb = beanb;
    }
    public void dosomething() {
        system.out.println("beana is doing something");
    }
}

beanb:

@component
public class beanb {
    private beana beana;
    public beanb() {
        system.out.println("beanb constructor");
    }
    @autowired
    public void setbeana(beana beana) {
        this.beana = beana;
    }
    public void dosomething() {
        system.out.println("beanb is doing something");
    }
}

运行结果

beana constructor
beanb constructor

setter 注入允许 spring 在实例化 bean 后再设置依赖,从而避免循环依赖问题。

方法 4:使用 objectfactory 或 provider 进行延迟注入

spring 提供了 objectfactoryprovider 接口,用于延迟获取 bean,从而避免循环依赖。

修改代码 beana:

@component
public class beana {
    private final objectfactory<beanb> beanbfactory;
    public beana(objectfactory<beanb> beanbfactory) {
        this.beanbfactory = beanbfactory;
        system.out.println("beana constructor");
    }
    public void dosomething() {
        beanb beanb = beanbfactory.getobject();
        system.out.println("beana is doing something with " + beanb);
    }
}

beanb:

@component
public class beanb {
    private final beana beana;
    @autowired
    public beanb(beana beana) {
        this.beana = beana;
        system.out.println("beanb constructor");
    }
    public void dosomething() {
        system.out.println("beanb is doing something");
    }
}

运行结果

beana constructor
beanb constructor

通过 objectfactory,beana 可以在需要时动态获取 beanb,避免了循环依赖。

方法 5:使用 @dependson 明确加载顺序

如果循环依赖是因为 bean 的加载顺序问题,可以使用 @dependson 指定加载顺序。

修改代码

beana:

@component
@dependson("beanb") // 指定 beanb 应该先加载
public class beana {
    @autowired
    private beanb beanb;
    public beana() {
        system.out.println("beana constructor");
    }
    public void dosomething() {
        system.out.println("beana is doing something");
    }
}

beanb:

@component
public class beanb {
    @autowired
    private beana beana;
    public beanb() {
        system.out.println("beanb constructor");
    }
    public void dosomething() {
        system.out.println("beanb is doing something");
    }
}

运行结果

beanb constructor
beana constructor

4. 总结

循环依赖是常见问题,但可以通过多种方式解决:

  • 使用 @lazy 延迟加载。
  • 重构代码,避免直接循环依赖。
  • 使用 setter 注入或 @postconstruct
  • 使用 objectfactory 或 provider 进行延迟注入。
  • 使用 @dependson 明确加载顺序。

推荐方案

如果可以重构代码,消除循环依赖本身是最佳实践。在必要时,结合 @lazyobjectfactory 来解决循环依赖问题。

到此这篇关于springboot循环依赖问题及其解决办法的文章就介绍到这了,更多相关springboot循环依赖内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2025  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com