SpringIoc源碼(十五)- BeanFactory(四)- getBean(doGetBean上 - 緩存中獲取)

目錄

下面爲非第一次getBean(已經初始化過一次),獲取的最短路徑

1、獲取真實的BeanName(傳入的可能是別名等)

     1)、判斷是否爲獲取FactoryBean

    2)、 判斷是否爲別名

2、從緩存中獲取實例(如果已經調用過getBean)

3、判斷返回真正獲取的對象

1、如果源名稱以&開頭,則判斷是否爲NullBean,判斷是否不存在(直接拋異常),返回工廠本身

2、獲取正常的Bean(非FactoryBean類型),則直接返回。

3、不是獲取FactoryBean本身,但是獲取的是FactoryBean類型的Bean

3-1)、從緩存中獲取(非第一次)

3-2)、獲取實例(第一次獲取)


    之所以先寫了ApplicationContext的refresh方法再寫繼續寫BeanFactory的getBean核心方法,是因爲覺得項目中根本不會直接使用BeanFactory肯定還是會使用ApplicationContext。並且每個Bean的生命週期會在getBean中完成,其中就會回調所有的BeanPostProcessorpostProcessBeforeInitializationpostProcessAfterInitialization方法。

    先梳理一下getBean的整個過程(只分析單利Bean)

    1、首先會從緩存中獲取;否則(第一次)需要根據之前注入的BeanDefinition進行創建。

    2、創建之前需要先處理ovverride和前置準備;在創建時會有依賴注入的情況(需要解決循環依賴的問題),則再分析真正的創建

    3、創建過程:創建Bean實例;記錄創建Bean的工廠;屬性注入;初始化Bean(Bean的生命週期在這裏完成,包括註冊DisposableBean)

 

下面爲非第一次getBean(已經初始化過一次),獲取的最短路徑

    開始:

@Override
public Object getBean(String name) throws BeansException {
    return doGetBean(name, null, null, false);
}
protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
    @Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {

    final String beanName = transformedBeanName(name);

    // Eagerly check singleton cache for manually registered singletons.
    Object sharedInstance = getSingleton(beanName);
    // 省略
}

1、獲取真實的BeanName(傳入的可能是別名等)

    調用getBean方法時候,傳入的字符串可能是別名,或者是FactoryBean類型則可能傳入&開頭的,以獲取FactoryBean本身。所以需要考慮各種情況,拿到真是的bean名稱。

protected String transformedBeanName(String name) {
    return canonicalName(BeanFactoryUtils.transformedBeanName(name));
}

     1)、判斷是否爲獲取FactoryBean

public static String transformedBeanName(String name) {
    if (!name.startsWith(BeanFactory.FACTORY_BEAN_PREFIX)) {
        return name;
    }
    return transformedBeanNameCache.computeIfAbsent(name, beanName -> {
        do {
            beanName = beanName.substring(BeanFactory.FACTORY_BEAN_PREFIX.length());
        }
        while (beanName.startsWith(BeanFactory.FACTORY_BEAN_PREFIX));
        return beanName;
    });
}

    判斷名稱是否以&開頭,不是則直接返回當前字符串;是則是獲取FactoryBean本身,則需要截取字符串,緩存獲取FactoryBean的名稱。緩存到BeanFactoryUtils

Map<String, String> transformedBeanNameCache

    2)、 判斷是否爲別名

public String canonicalName(String name) {
    String canonicalName = name;
    String resolvedName;
    do {
        resolvedName = this.aliasMap.get(canonicalName);
        if (resolvedName != null) {
            canonicalName = resolvedName;
        }
    }
    while (resolvedName != null);
    return canonicalName;
}

  當前字符串要不是Bean名稱本身,要不是就是別名。那麼直接在別名中獲取一下,有則返回別名,否則返回Bean名稱本身。

 

2、從緩存中獲取實例(如果已經調用過getBean)

@Override
@Nullable
public Object getSingleton(String beanName) {
    return getSingleton(beanName, true);
}
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
    Object singletonObject = this.singletonObjects.get(beanName);
    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);
                }
            }
        }
    }
    return singletonObject;
}

    如果第一次調用getBean初始化完成後,則會將實例化的Bean放入DefaultSingletonBeanRegistrysingletonObjects容器中;爲了解決單利循環依賴的問題,則在依賴注入時,會提前將未完成的Bean提前放入earlySingletonObjects容器中。創建Bean的工廠與Bean名稱的關係存儲在singletonFactories容器中。

  1. 先直接先看是否初始化完成,如果沒有拿到,則判斷是否正在創建中。
  2. 如果是則從earlySingletonObjects中進行獲取。並且當前allowEarlyReference參數爲true,則如果正在依賴注入創建中,直接從工廠中獲取
  3. 如果工廠已經存在,則直接getBean調用工廠的getBean方法,並且提前暴露放到earlySingletonObjects中。

3、判斷返回真正獲取的對象

if (sharedInstance != null && args == null) {
    // 省略日誌
    bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
}

    上面獲取到的可能就是需要返回的Bean,但是當前可能需要獲取的是FactoryBean本身(是否或者FactoryBean已經緩存到BeanFactoryUtilstransformedBeanNameCache中)。獲取時傳入了上面獲取的實例,源名稱,Bean名稱,RootBeanDefinition爲null。

@Override
protected Object getObjectForBeanInstance(
        Object beanInstance, String name, String beanName, @Nullable RootBeanDefinition mbd) {

    String currentlyCreatedBean = this.currentlyCreatedBean.get();
    if (currentlyCreatedBean != null) {
        registerDependentBean(beanName, currentlyCreatedBean);
    }

    return super.getObjectForBeanInstance(beanInstance, name, beanName, mbd);
}

     判斷當前是否正在有依賴注入的Bean沒有初始化完成(非第一次調用則肯定已經初始化完成了),在繼續調用父類AbstractBeanFactory的方法。

protected Object getObjectForBeanInstance(
        Object beanInstance, String name, String beanName, @Nullable RootBeanDefinition mbd) {

    // Don't let calling code try to dereference the factory if the bean isn't a factory.
    if (BeanFactoryUtils.isFactoryDereference(name)) {
        if (beanInstance instanceof NullBean) {
            return beanInstance;
        }
        if (!(beanInstance instanceof FactoryBean)) {
            throw new BeanIsNotAFactoryException(beanName, beanInstance.getClass());
        }
        if (mbd != null) {
            mbd.isFactoryBean = true;
        }
        return beanInstance;
    }

    // Now we have the bean instance, which may be a normal bean or a FactoryBean.
    // If it's a FactoryBean, we use it to create a bean instance, unless the
    // caller actually wants a reference to the factory.
    if (!(beanInstance instanceof FactoryBean)) {
        return beanInstance;
    }

    Object object = null;
    if (mbd != null) {
        mbd.isFactoryBean = true;
    }
    else {
        object = getCachedObjectForFactoryBean(beanName);
    }
    if (object == null) {
        // Return bean instance from factory.
        FactoryBean<?> factory = (FactoryBean<?>) beanInstance;
        // Caches object obtained from FactoryBean if it is a singleton.
        if (mbd == null && containsBeanDefinition(beanName)) {
            mbd = getMergedLocalBeanDefinition(beanName);
        }
        boolean synthetic = (mbd != null && mbd.isSynthetic());
        object = getObjectFromFactoryBean(factory, beanName, !synthetic);
    }
    return object;
}

1、如果源名稱以&開頭,則判斷是否爲NullBean,判斷是否不存在(直接拋異常),返回工廠本身

2、獲取正常的Bean(非FactoryBean類型),則直接返回。

3、不是獲取FactoryBean本身,但是獲取的是FactoryBean類型的Bean

 

3-1)、從緩存中獲取(非第一次)

getCachedObjectForFactoryBean(beanName);

    則在AbstractBeanFactory的父類FactoryBeanRegistrySupport,的factoryBeanObjectCache容器中獲取

@Nullable
protected Object getCachedObjectForFactoryBean(String beanName) {
    return this.factoryBeanObjectCache.get(beanName);
}

3-2)、獲取實例(第一次獲取)

    那麼上面獲取到的beanInstance爲FactoryBean,先或者合併的BeanDefinition,再判斷是否爲synthetic(合成的)。

protected Object getObjectFromFactoryBean(FactoryBean<?> factory, String beanName, boolean shouldPostProcess) {
    // 判斷爲單利
    if (factory.isSingleton() && containsSingleton(beanName)) {
        synchronized (getSingletonMutex()) {
            // 從緩存中判斷
            Object object = this.factoryBeanObjectCache.get(beanName);
            if (object == null) {
                // 直接用FactoryBean中調用getObject方法獲取,或者判斷是否正在創建中
                object = doGetObjectFromFactoryBean(factory, beanName);
                // Only post-process and store if not put there already during getObject() call above
                // (e.g. because of circular reference processing triggered by custom getBean calls)
                Object alreadyThere = this.factoryBeanObjectCache.get(beanName);
                if (alreadyThere != null) {
                    object = alreadyThere;
                }
                else {
                    if (shouldPostProcess) {
                        if (isSingletonCurrentlyInCreation(beanName)) {
                            // Temporarily return non-post-processed object, not storing it yet..
                            return object;
                        }
                        beforeSingletonCreation(beanName);
                        try {
                            object = postProcessObjectFromFactoryBean(object, beanName);
                        }
                        catch (Throwable ex) {
                            throw new BeanCreationException(beanName,
                                    "Post-processing of FactoryBean's singleton object failed", ex);
                        }
                        finally {
                            afterSingletonCreation(beanName);
                        }
                    }
                    if (containsSingleton(beanName)) {
                        // 放入緩存中,第二次就從緩存中獲取
                        this.factoryBeanObjectCache.put(beanName, object);
                    }
                }
            }
            return object;
        }
    }
    // 省略非單利
}

    先用synchronized鎖住singletonObjects,又從緩存中獲取一遍。

private Object doGetObjectFromFactoryBean(final FactoryBean<?> factory, final String beanName)
        throws BeanCreationException {

    Object object;
    try {
        if (System.getSecurityManager() != null) {
            AccessControlContext acc = getAccessControlContext();
            try {
                object = AccessController.doPrivileged((PrivilegedExceptionAction<Object>) factory::getObject, acc);
            }
            catch (PrivilegedActionException pae) {
                throw pae.getException();
            }
        }
        else {
            object = factory.getObject();
        }
    }
    catch (FactoryBeanNotInitializedException ex) {
        throw new BeanCurrentlyInCreationException(beanName, ex.toString());
    }
    catch (Throwable ex) {
        throw new BeanCreationException(beanName, "FactoryBean threw exception on object creation", ex);
    }

    // Do not accept a null value for a FactoryBean that's not fully
    // initialized yet: Many FactoryBeans just return null then.
    if (object == null) {
        if (isSingletonCurrentlyInCreation(beanName)) {
            throw new BeanCurrentlyInCreationException(
                    beanName, "FactoryBean which is currently in creation returned null from getObject");
        }
        object = new NullBean();
    }
    return object;
}

     否則直接從FactoryBean中獲取,但是可能或者不到,則判斷是否正在創建中,則後面從創建中獲取。主要的方法就是FactoryBean#getObject方法。第一次獲取,則也會執行FactoryBean類型的Bean的生命週期的 回調BeanPostProcess的方法。比如之前修改jpa的JpaRepository子類(通過JpaRepositoryFactoryBean#getObject創建)就有該回調。如下:

@Override
protected Object postProcessObjectFromFactoryBean(Object object, String beanName) {
    return applyBeanPostProcessorsAfterInitialization(object, beanName);
}

    但是隻回調了所有BeanPostProcessorpostProcessAfterInitialization方法:

@Override
public Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName)
        throws BeansException {

    Object result = existingBean;
    for (BeanPostProcessor processor : getBeanPostProcessors()) {
        Object current = processor.postProcessAfterInitialization(result, beanName);
        if (current == null) {
            return result;
        }
        result = current;
    }
    return result;
}

    最後將FactoryBean通過getObject方法,獲取到的對象放入factoryBeanObjectCache緩存,第二次就可以從該容器中獲取了

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章