V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
jintianfengda
V2EX  ›  Java

请教,关于 springboot 循环依赖的问题

  •  
  •   jintianfengda · 2021-01-06 11:29:28 +08:00 · 1613 次点击
    这是一个创建于 1436 天前的主题,其中的信息可能已经有所发展或是发生改变。
    我记得之前 spring 如果使用 Autowired 注解注入成员变量的话,是会启动的时候抛错提示创建 bean 失败的,但是我刚测了一下 2.3.3 版本下如果 A 注入 B,B 注入 A 是不会引起这个错误的,我也并没有使用延迟加载,这是新版本的特性吗,还是我测试的姿势不对
    InkAndBanner
        1
    InkAndBanner  
       2021-01-06 11:44:26 +08:00
    不会报错吗?
    yukong
        2
    yukong  
       2021-01-06 11:48:55 +08:00
    scope 如果是 Singleton 循环依赖不会保存的 如果是 prototype 就会报错
    chendy
        3
    chendy  
       2021-01-06 11:50:23 +08:00
    不是 spring-boot,是 spring
    循环依赖的情况,字段注入和 setter 注入可以正常工作,构造器注入无法工作
    sleaves
        4
    sleaves  
       2021-01-06 12:12:28 +08:00
    楼上正解
    jintianfengda
        5
    jintianfengda  
    OP
       2021-01-06 12:17:21 +08:00
    @chendy 感谢,这么多年一直记错了
    Joker123456789
        6
    Joker123456789  
       2021-01-06 14:36:34 +08:00   ❤️ 1
    默认是单例啊,单例是不存在循环依赖问题的。

    创建一个 Bean 分为创建 对象实例,注入属性两部分。

    创建对象实例是没限制的,随便 new,问题在于第二步。

    spring 用了缓存来解决的。

    ----------------------------------------------------------

    1. 当创建 A 以后,会把 A 放入缓存中,然后往 A 注入 B [此时 A 创建成功,但是还没把 B 注入进来] ,

    2. 往 A 注入 B 的是时候 需要 B 的实例,它会先去缓存获取 B,发现 B 不存在就会创建一个 B [此时 B 还只是创建成功,并没把 A 注入进来]

    3. B 创建完以后,要往 B 注入 A,同样需要 A 的实例,所以先去缓存获取 A,发现缓存中存在 A (因为第一步已经建好了), 就会从缓存中取出来然后注入到 B,并把 B 放入缓存 [此时 B 变成了一个被注入了的完成对象] ,

    4. 将 B 注入到 A [此时 A 变成了一个被注入了的完整对象]

    5. 以上就是 spring 解决单例模式下的循环依赖问题
    jintianfengda
        7
    jintianfengda  
    OP
       2021-01-06 14:42:55 +08:00
    @Joker123456789 感谢讲解,spring 初始化 bean 的三层缓存的方式我晓得,就是我印象里记得构造器方式和字段方式注入都会提示循环依赖了,是我记错了,也是我对原理理解的不够透彻,只是表面了解
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   3131 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 22ms · UTC 04:50 · PVG 12:50 · LAX 20:50 · JFK 23:50
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.