每天一点点,Spring执行创建bean(AbstractAutowireCapableBeanFactory#doCreateBean)

Spring 创建bean流程 AbstractAutowireCapableBeanFactory#doCreateBean

1、 判断当前bean是不是单例的,如果是单例的,先从缓存中取,如果存在直接返回(在创建之前已经由依赖注入已生成),如果不存在执行创建

if (mbd.isSingleton()) {
	instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);
}
if (instanceWrapper == null) {
	instanceWrapper = createBeanInstance(beanName, mbd, args);
}

2、如果不是NullBean对象,进行缓存

if (beanType != NullBean.class) {
	mbd.resolvedTargetType = beanType;
}

3、判断缓存是否已经处理过MergedBeanPostProcessors,如果没有则执行

synchronized (mbd.postProcessingLock) {
	if (!mbd.postProcessed) {
		try {
			applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
		}
		mbd.postProcessed = true;
	}
}

4、判断是否允许循环依赖,如果支持将bean的创建暴露,存在则存到DefaultSingletonBeanRegistry#singletonFactories中

boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
	addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}

5、更换一个变量,执行后面的逻辑。
5.1、分析 :此处为什么需要重新定义一个变量,而不是直接使用产生的对象 ??
我的理解 : 因为上面在判断允许循环依赖的时候,加入的是原始bean对象,在循环依赖的时候,如果需要的对象是需要代理的,则被代理的对象就是这个原始对象bean。如果 此处不重新更换一个变量exposedObject进行后续的处理的话,就有可能导致 在循环依赖的时候里面的被代理对象bean和 exposedObject 对象不一致,这样是有问题的

Object exposedObject = bean;

6、执行属性填充

populateBean(beanName, mbd, instanceWrapper); 

7、执行初始化

exposedObject = initializeBean(beanName, exposedObject, mbd);

8、如果支持循环依赖
8.1、取出二级缓存中的早期暴露的对象earlySingletonReference ,如果存在则代表存在循环依赖

//如果earlySingletonReference != null 则代表当前bean存在循环依赖
Object earlySingletonReference = getSingleton(beanName, false);

8.2、如果 earlySingletonReference != null 代表存在循环依赖
8.2.1 判断当前暴露的类是不是和生成的bean一致,如果一致修改当前类的实例为二级缓存中的bean对象

if (exposedObject == bean) {
// 因为当前ben可能已经被代理了,所以放在单例池的bean应该是代理对象
	exposedObject = earlySingletonReference;
}

9、如果bean有定义销毁方法,注册这些bean

try {
	registerDisposableBeanIfNecessary(beanName, bean, mbd);
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值