美文网首页技术分享
Spring循环依赖导致的@PostConstruct顺序没有按

Spring循环依赖导致的@PostConstruct顺序没有按

作者: 愤怒的老照 | 来源:发表于2021-05-25 20:26 被阅读0次

    背景

    bean的生命周期如下图所示:

    image.png

    @PostConstruct注解是会被一个专门的BeanPostProcessor接口的具体实现类来处理的,实现类是:InitDestroyAnnotationBeanPostProcessor。

    按照加载顺序,@PostConstruct也会按照依赖顺序执行。

    但是在代码里并没有按照期望顺序执行,依赖关系如下:

    @Service
    @Slf4j
    public class A {
        @Resource
        private B b;
    
        @PostConstruct
        public void init(){
            log.info("A PostConstruct");
            b.test();
        }
    }
    
    
    
    @Service
    @Slf4j
    public class B {
        @Resource
        private C c;
    
        @PostConstruct
        public void init(){
            log.info("B PostConstruct");
        }
    
        public void test(){
            log.info("B Test");
        }
    }
    
    
    @Service
    public class C {
    
        @Resource
        private A a;
    }
    

    A对象里依赖了B对象,并且A的@PostConstruct方法依赖了B的@PostConstruct生成的数据,但是A的@PostConstruct,导致拿到的数据为空

    问题分析

    经过debug,发现是由于循环依赖导致,B先加载并且提前暴露,导致A执行@PostConstruct时B还没有初始化完成

    为解决循环依赖,spring采用三级缓存机制,将实例化成功的对象加载到三级缓存,

    这三级缓存的作用分别是:

    singletonFactories : 进入实例化阶段的单例对象工厂的cache (三级缓存)
    
    earlySingletonObjects :完成实例化但是尚未初始化的,提前暴光的单例对象的Cache (二级缓存)
    
    singletonObjects:完成初始化的单例对象的cache(一级缓存)
    

    我们在创建bean的时候,会首先从cache中获取这个bean,这个缓存就是sigletonObjects。主要的调用方法是:

    protected Object getSingleton(String beanName, boolean allowEarlyReference) {
        Object singletonObject = this.singletonObjects.get(beanName);
        //isSingletonCurrentlyInCreation()判断当前单例bean是否正在创建中
        if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
            synchronized (this.singletonObjects) {
                singletonObject = this.earlySingletonObjects.get(beanName);
                //allowEarlyReference 是否允许从singletonFactories中通过getObject拿到对象
                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);
                    }
                }
            }
        }
        return (singletonObject != NULL_OBJECT ? singletonObject : null);
    }
    

    从上面三级缓存的分析,我们可以知道,Spring解决循环依赖的诀窍就在于singletonFactories这个三级cache。这个cache的类型是ObjectFactory,定义如下:

    public interface ObjectFactory<T> {
        T getObject() throws BeansException;
    }
    

    这个接口在AbstractBeanFactory里实现,并在核心方法doCreateBean()引用下面的方法:

    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);
            }
        }
    }
    

    这段代码发生在createBeanInstance之后,populateBean()之前,也就是说单例对象此时已经被创建出来(调用了构造器)。这个对象已经被生产出来了,此时将这个对象提前曝光出来

    所以在发生循环依赖时,B还未初始化,所以@PostConstruct方法还未执行

    解决方案

    • 解决循环依赖

    • 将逻辑从@PostConstruct里抽出来

    相关文章

      网友评论

        本文标题:Spring循环依赖导致的@PostConstruct顺序没有按

        本文链接:https://www.haomeiwen.com/subject/pxicsltx.html