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的单例对象的初始化主要分为三步:

  1. createBeanInstance:实例化,其实也就是调用对象的构造方法实例化对象
  2. populateBean:填充属性,这一步主要是多bean的依赖属性进行填充
  3. 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 为何需要三级缓存解决循环依赖,而不是二级缓存?

面试题:Spring 为何需要三级缓存解决循环依赖,而不是二级缓存?

Spring三级缓存

Spring循环依赖 | Spring三级缓存 | 看完必有收获