Spring循环依赖源码剖析
一、场景介绍

二、整理执行流程总结


三、源码分析
- 编写测试类
/** * 测试循环依赖 */ @Test public void testCyclicDependence(){ ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext-cyclicDependence.xml"); ABean aBean = applicationContext.getBean(ABean.class); aBean.print(); } - 在类Spring容器DefaultListableBeanFactory类中定位到方法preInstantiateSingletons(),设置断点(此方法进行Bean的实例化操作)

- 在此方法中,定位到实例化Bean的核心方法getBean(beanName),设置断点,进行跳转,并进入此方法中

- 定位到getSingleton(beanName)方法,并断点进入此方法

-
从一级缓存中获取BeanA,获取不到

-
将断点设置到类AbstractBeanFactory中,定位到创建单例bean的方法getSingleton(),进入此方法



getSingleton()方法最终执行createBean方法,进行bean的创建,因此继续进入此方法

- 定位到doCreateBean方法,并进入

实例化BeanA

继续向下执行,加入到spring的三级缓存,是一个工厂

将BeanA的工厂对象存储到三级缓存

BeanA属性填充


BeanA属性填充BeanB





此处流程和创建BeanA的流程一致





















总结:
二级缓存:复用(如果BeanA中有多个属性bean都依赖BeanB,此时就可以直接从二级缓存中取即可)
三级缓存:AOP增强
对象单一原则:不建议在一个集合中存储两种状态的对象,一级缓存存储走完spring生命周期的bean,二级缓存存储未走完的对象,三级缓存存储bean的工厂对象,并且可以生产通过AOP增强的对象
BeanA存在循环依赖:三级缓存--->二级缓存--->一级缓存
BeanB不存在循环依赖:三级缓存--->一级缓存
3万+





