spring的循环依赖及使用三级缓存解决循环依赖;注入bean到spring容器中
Posted 好大的月亮
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring的循环依赖及使用三级缓存解决循环依赖;注入bean到spring容器中相关的知识,希望对你有一定的参考价值。
循环依赖
循环依赖其实就是循环引用,也就是两个或则两个以上的bean互相持有对方,最终形成闭环。比如A依赖于B,B依赖于C,C又依赖于A。
Spring中循环依赖场景有:
(1)构造器的循环依赖
(2)field属性的循环依赖。
spring对循环依赖的解决(三级缓存)
Spring的循环依赖的理论依据其实是基于Java的引用传递,当我们获取到对象的引用时,对象的field或则属性是可以延后设置的(但是构造器必须是在获取引用之前)。
Spring的单例对象的初始化主要分为三步:
- createBeanInstance:实例化,其实也就是调用对象的构造方法实例化对象
- populateBean:填充属性,这一步主要是多bean的依赖属性进行填充
- initializeBean:调用spring xml中的init 方法。
循环依赖主要发生在第一、第二部。也就是构造器循环依赖和field循环依赖;
对于单例来说,在Spring容器整个生命周期内,有且只有一个对象,所以很容易想到这个对象应该存在Cache中,Spring为了解决单例的循环依赖问题,使用了三级缓存。
这三级缓存分别指:
singletonFactories
: 单例对象工厂的cache
earlySingletonObjects
:提前暴光的单例对象的Cache
singletonObjects
:单例对象的cache
org.springframework.beans.factory.support.DefaultSingletonBeanRegistry
三级缓存顺序
spring三级缓存不能解决构造器的循环依赖
创建中的对象提前曝光到三级缓存中,发生在bean创建之后还没被依赖注入;
文字流程:
A的某个field或者setter依赖了B的实例对象,同时B的某个field或者setter依赖了A的实例对象”这种循环依赖的情况。A首先完成了初始化的第一步,并且将自己提前曝光到singletonFactories中,此时进行初始化的第二步,发现自己依赖对象B,此时就尝试去get(B),发现B还没有被create,所以走create流程,B在初始化第一步的时候发现自己依赖了对象A,于是尝试get(A),尝试一级缓存singletonObjects(肯定没有,因为A还没初始化完全),尝试二级缓存earlySingletonObjects(也没有),尝试三级缓存singletonFactories,由于A通过ObjectFactory将自己提前曝光了,所以B能够通过ObjectFactory.getObject拿到A对象(虽然A还没有初始化完全,但是总比没有好呀),B拿到A对象后顺利完成了初始化阶段1、2、3,完全初始化之后将自己放入到一级缓存singletonObjects中。此时返回A中,A此时能拿到B的对象顺利完成自己的初始化阶段2、3,最终A也完成了初始化,进去了一级缓存singletonObjects中,而且更加幸运的是,由于B拿到了A的对象引用,所以B现在hold住的A对象完成了初始化。
看完就知道因为加入singletonFactories三级缓存的前提是执行了构造器,所以构造器的循环依赖没法解决。
参考了大佬的博文
https://blog.csdn.net/u010853261/article/details/77940767
注入bean到spring容器中
1-@Bean@Service等注解
2-手动注册
2-1 通过GenericBeanDefinition注册
public static void main(String[] args) {
DefaultListableBeanFactory context =
new DefaultListableBeanFactory();
//在此构造bean定义
GenericBeanDefinition gbd = new GenericBeanDefinition();
//Message要被注入的class
gbd.setBeanClass(Message.class);
//设置属性注入
MutablePropertyValues mpv = new MutablePropertyValues();
//message属性名
mpv.add("message", "this is a bean");
//注册到环境上下文
context.registerBeanDefinition("myBean", gbd);
Message bean = context.getBean(Message.class);
bean.print();
Message myBean = (Message) context.getBean("myBean");
myBean.print();
}
2-2 通过BeanDefinitionBuilder注册和GenericBeanDefinition差不多的
public static void main(String[] args) {
DefaultListableBeanFactory context =
new DefaultListableBeanFactory();
//用到了构建者模式
BeanDefinitionBuilder b =
BeanDefinitionBuilder.rootBeanDefinition(Message.class)
.addPropertyValue("message", "this is a bean");
context.registerBeanDefinition("myBean", b.getBeanDefinition());
Message bean = context.getBean(Message.class);
bean.print();
Message myBean = (Message) context.getBean("myBean");
myBean.print();
}
2-3 通过BeanFactoryPostProcessor
public class MessageBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
@Override
public void postProcessBeanFactory(
ConfigurableListableBeanFactory beanFactory) throws BeansException {
// TODO Auto-generated method stub
GenericBeanDefinition gbd = new GenericBeanDefinition();
gbd.setBeanClass(Message.class);
MutablePropertyValues mpv = new MutablePropertyValues();
mpv.add("message", "this is a BeanFactoryPostProcessor bean");
gbd.setPropertyValues(mpv);
((DefaultListableBeanFactory) beanFactory).registerBeanDefinition(
"myBean", gbd);
}
}
2-4 通过BeanDefinitionRegistryPostProcessor
public class MessageBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {
@Override
public void postProcessBeanFactory(
ConfigurableListableBeanFactory beanFactory) throws BeansException {
// TODO Auto-generated method stub
}
@Override
public void postProcessBeanDefinitionRegistry(
BeanDefinitionRegistry registry) throws BeansException {
// TODO Auto-generated method stub
GenericBeanDefinition gbd = new GenericBeanDefinition();
gbd.setBeanClass(Message.class);
MutablePropertyValues mpv = new MutablePropertyValues();
mpv.add("message", "this is a BeanDefinitionRegistryPostProcessor bean");
gbd.setPropertyValues(mpv);
registry.registerBeanDefinition(
"myBean", gbd);
}
}
2-5 BeanFactoryAware
@Component
public class LocationRegister implements BeanFactoryAware {
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
DefaultListableBeanFactory listableBeanFactory = (DefaultListableBeanFactory) beanFactory;
//方式1 Location location = new Location();
// listableBeanFactory.registerSingleton(Location.class.getName(),location);
//方式2 BeanDefinition locationBeanDefinition = new RootBeanDefinition(Location.class);
// listableBeanFactory.registerBeanDefinition(Location.class.getName(),locationBeanDefinition);
}
}
参考了大佬的博文
https://www.cnblogs.com/dongguangming/p/12792789.html
以上是关于spring的循环依赖及使用三级缓存解决循环依赖;注入bean到spring容器中的主要内容,如果未能解决你的问题,请参考以下文章
手撕Spring源码,详细理解Spring循环依赖及三级缓存
面试题:Spring 为何需要三级缓存解决循环依赖,而不是二级缓存?
面试题:Spring 为何需要三级缓存解决循环依赖,而不是二级缓存?