天天看点

spring处理对象相互依赖注入的问题

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/w1lgy/article/details/81086171

最近重新再看spring的内容的时候有个疑问:当spring对象对象存在相互依赖注入的时候,会不会造成死循环,如果不会spring是怎么解决的?带着这个疑问我自己在eclipse试了一下,发现是能正常运行的,那么spring是怎么做到的呢,带着疑问我百度了一下,查到一篇博客可供大家参考:

https://blog.csdn.net/u012410733/article/details/61062858 。但是这篇博客并不能完全解决的疑问,于是我又把问题发给了线上培训的导师和培训的会员群上,经过老师的解读加上我查阅资料,做出以下总结:

一、Spring 循环依赖需要满足三个条件:

1、setter 注入(构造器注入不能循环依赖)

2、singleton bean(非单例 bean 不支持)

3、AbstractAutowireCapableBeanFactory#allowCircularReferences 这个 boolean 属性控制是否可以循环,默认为 true。

二、Spring 创建对象分为三个过程:

1、创建对象实例 Object obj = new Object() 或者 Object obj = new Object(xxx);

AbstractAutowireCapableBeanFactory#createBeanInstance

2、依赖注入: obj.setXxx(xxx) {多个属性就是 foreach}

AbstractAutowireCapableBeanFactory#populateBean

3、Spring bean 扩展方法:init-method,BeanPostProcess,XXXAware 扩展

AbstractAutowireCapableBeanFactory#initializeBean

实现原理:

我们知道在循环依赖当中其实就是多个 bean 调用 BeanFactory#getBean

具体实现在 org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean

而这个方在 org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#doCreateBean 方法前。

首先来看一下 AbstractBeanFactory#doGetBean

在进入 doGetBean 方法时就会从缓存里面拿 beanName 对应的对象。

其实中就包含创建的 ObjectFactory

总结

相互依赖:A 依赖 B, B 依赖 A

当 A 创建的时候,会把 A 对应的 ObjectFactory 放在缓存中,当依赖注入的时候发现了 B 对象,调用 getBean() 方法获取 B 对象, 然后创建 B 对象,会把 B 对应的 ObjectFactory 放在缓存中。

此时 B 依赖 A ,然后再调用 getBean 获取 A 对象, 此时调用 AbstractBeanFactory#doGetBean 从缓存中获取到 A 对应的 ObjectFactory。

这样就避免了死循环,然后再创建成功之后删除 ObjectFactory 完成依赖注入。

思路:中间对象去解决循环依赖。

继续阅读