【Spring】三、SpringIOC Bean的初始化流程-循环依赖

本文详细剖析了Spring中Bean循环依赖的原理,通过三级缓存(singletonObjects, earlySingletonObjects, singletonFactories)解释创建过程,并揭示了setter注入支持但构造方法不支持的特性。深入理解如何在Spring IoC容器中处理此类问题。
摘要由CSDN通过智能技术生成

上一篇: 【Spring】二、SpringIOC Bean的初始化流程

一、循环依赖现象理解

  • Spring bean的创建,本质上是一个对象的创建;
  • 一个完整的对象包含两部分:当前对象和对象的属性所依赖的另一个对象;
    对象实例化通过反射实现,而对象的属性实例化通过依赖注入实现;
  • Spring实例化bean是通过ApplicationContext.getBean()方法来进行的,如果要获取的对象又依赖了另一个对象,那么其首先会创建当前对象,然后通过递归调用ApplicationContext.getBean()方法来获取所依赖的对象,最后将获取到的对象注入到当前对象中;

二、模拟与理解分析

1、三级缓存

spring内部有三级缓存:

  • singletonObjects 一级缓存,用于保存实例化、注入、初始化完成的bean实例。
  • earlySingletonObjects 二级缓存,用于保存实例化完成的bean实例。
  • singletonFactories 三级缓存,用于保存bean创建工厂,以便于后面扩展有机会创建代理对象。

2、其余缓存

在这里插入图片描述

在这里插入图片描述

3、流程梳理

  • 现在有两个bean对象 MyBean、 HeBean

  • 前提:(HeBean 依赖 MyBean,MyBean依赖HeBean)

  • 假设容器首先创建HeBean:

  1、创建HeBean对象:反射 (此时MyBean还没有创建)
  
  2、把“heBean”放入 singletonFactories的Map中;
  
   protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
		Assert.notNull(singletonFactory, "Singleton factory must not be null");
		synchronized (this.singletonObjects) {
			if (!this.singletonObjects.containsKey(beanName)) {
				this.singletonFactories.put(beanName, singletonFactory);
				this.earlySingletonObjects.remove(beanName);
				this.registeredSingletons.add(beanName);
			}
		}
	}
  
  3、给heBean设置属性值(也就是给heBean的成员变量赋值)
  
  4、当HeBean依赖了MyBean,而MyBean还没有创建,此时HeBean无法完成属性赋值,HeBean还  是一个半成品,然后就跳到了:beanFactory.getBean(“myBean”); (递归到了获取bean的入口处)
  
  5、开始创建MyBean对象
      5.1、创建MyBean对象:反射 (此时HeBean是个半成品)
      
      5.2、把“myBean”放入 singletonFactories的Map
       	protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
		Assert.notNull(singletonFactory, "Singleton factory must not be null");
		synchronized (this.singletonObjects) {
			if (!this.singletonObjects.containsKey(beanName)) {
				this.singletonFactories.put(beanName, singletonFactory);
				this.earlySingletonObjects.remove(beanName);
				this.registeredSingletons.add(beanName);
			}
		}
	}
       
      5.3、给myBean设置属性值(也就是给MyBean的成员变量赋值)
      5.4MyBean此时是一个半成品,此时代码:
        //TODO 解析候选者bean,会调beanFactory.getBean()方法获取bean,该方法内部会进                  行初始化
        instanceCandidate = descriptor.resolveCandidate(autowiredBeanName, type, this);
       // 并没有完成heBean属性设置然后就跳到了:beanFactory.getBean(“heBean”);(递归到了获取bean的入口处)
        //去入口处:
       public Object resolveCandidate(String beanName, Class<?> requiredType,                  BeanFactory beanFactory) throws BeansException {
		  // TODO 根据bean名称获取bean对象
		  return beanFactory.getBean(beanName);
	  }
      5.5、从这里获取heBean: 
         ObjectFactory<?> singletonFactory =
         this.singletonFactories.get(beanName);
         singletonObject = singletonFactory.getObject();
         
  6、把heBean放入earlySingletonObjects的map中
   @Nullable
	protected Object getSingleton(String beanName, boolean allowEarlyReference) {
		//从singletonObjects这个Map中获取bean对象,如果存在,则不需要创建
		Object singletonObject = this.singletonObjects.get(beanName);
		// 如果缓存中不存在目标对象,则判断当前对象是否已经处于创建过程中,在前面的第一次尝试获取该对象
		// 的实例之后,就会将该对象标记为正在创建中,因而此处尝试获取该对象的时候,if判断就会为true
		if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {

			synchronized (this.singletonObjects) {
				singletonObject = this.earlySingletonObjects.get(beanName);

				if (singletonObject == null && allowEarlyReference) {
					ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
					if (singletonFactory != null) {
						singletonObject = singletonFactory.getObject();
						this.earlySingletonObjects.put(beanName, singletonObject);
						this.singletonFactories.remove(beanName);
					}
				}
			}
		}
		//返回单例bean对象
		return singletonObject;
	}
    
   7、回到入口处:
      public Object resolveCandidate(String beanName, Class<?> requiredType, BeanFactory beanFactory) throws BeansException {
		// TODO 根据bean名称获取bean对象
		return beanFactory.getBean(beanName);
	}
   8、此时已经拿到了HeBean对象,反射调用:field.set(bean, value);MyBean对象的heBean属性赋值,此时MyBean对象就完全创建好了;
   
   9、把myBean放入ioc容器:
      	/**
	 * Add the given singleton object to the singleton cache of this factory.
	 * <p>To be called for eager registration of singletons.
	 * @param beanName the name of the bean
	 * @param singletonObject the singleton object
	 */
	protected void addSingleton(String beanName, Object singletonObject) {

		synchronized (this.singletonObjects) {

			//TODO bean保存在singletonObjects的Map中,key为bean的名称,value为bean对象实例
			this.singletonObjects.put(beanName, singletonObject);
			this.singletonFactories.remove(beanName);
			this.earlySingletonObjects.remove(beanName);
			this.registeredSingletons.add(beanName);
		}
	}
   10、返回到 resolveCandidate(String beanName, Class<?> requiredType, BeanFactory beanFactory) 方法
   (相当于返回到创建HeBean的第4步,此时已经拿到mybean对象,通过反射调用field.set(bean, value);
   把heBean对象的mybean属性赋值,此时HeBean对象也就创建好了,放入ioc的map容器中)

4、循环依赖支持条件:

1、setter注入 支持
2、构造方法注入 不支持
3、bean的多例模式 不支持

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值