Spring IoC源码解析之getBean(2)

* 方法参数typeCheckOnly ,是用来判断调用getBean(…) 方法时,表示是否为仅仅进行类型检查获取Bean对象

* 如果不是仅仅做类型检查,而是创建Bean对象,则需要调用markBeanAsCreated(String beanName) 方法,进行记录

*/

if (!typeCheckOnly) {

markBeanAsCreated(beanName);

}

try {

//从容器中获取beanName相应的GenericBeanDefinition对象,并将其转换为RootBeanDefinition对象

final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);

//检查当前创建的bean定义是不是抽象的bean定义

checkMergedBeanDefinition(mbd, beanName, args);

//处理dependsOn的依赖(这个不是我们所谓的循环依赖 而是bean创建前后的依赖)

//依赖bean的名称

String[] dependsOn = mbd.getDependsOn();

if (dependsOn != null) {

for (String dep : dependsOn) {

//beanName是当前正在创建的bean,dep是正在创建的bean的依赖的bean的名称

if (isDependent(beanName, dep)) {

throw new BeanCreationException(mbd.getResourceDescription(), beanName,

“Circular depends-on relationship between '” + beanName + “’ and '” + dep + “'”);

}

//保存的是依赖beanName之间的映射关系:依赖beanName -> beanName的集合

registerDependentBean(dep, beanName);

try {

//获取dependsOn的bean

getBean(dep);

}

catch (NoSuchBeanDefinitionException ex) {

throw new BeanCreationException(mbd.getResourceDescription(), beanName,

“'” + beanName + “’ depends on missing bean '” + dep + “'”, ex);

}

}

}

//是单例则创建单例Bean

if (mbd.isSingleton()) {

//把beanName和一个singletonFactory匿名内部类传入用于回调

sharedInstance = getSingleton(beanName, () -> {

try {

//创建bean的逻辑

return createBean(beanName, mbd, args);

}

catch (BeansException ex) {

//创建bean的过程中发生异常,需要销毁关于当前bean的所有信息

destroySingleton(beanName);

throw ex;

}

});

bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);

}

else if (mbd.isPrototype()) {

// It’s a prototype -> create a new instance.

Object prototypeInstance = null;

try {

beforePrototypeCreation(beanName);

prototypeInstance = createBean(beanName, mbd, args);

}

finally {

afterPrototypeCreation(beanName);

}

bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);

}

else {

String scopeName = mbd.getScope();

final Scope scope = this.scopes.get(scopeName);

if (scope == null) {

throw new IllegalStateException(“No Scope registered for scope name '” + scopeName + “'”);

}

try {

Object scopedInstance = scope.get(beanName, () -> {

beforePrototypeCreation(beanName);

try {

return createBean(beanName, mbd, args);

}

finally {

afterPrototypeCreation(beanName);

}

});

bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);

}

catch (IllegalStateException ex) {

throw new BeanCreationException(beanName,

“Scope '” + scopeName + "’ is not active for the current thread; consider " +

“defining a scoped proxy for this bean if you intend to refer to it from a singleton”,

ex);

}

}

}

catch (BeansException ex) {

cleanupAfterBeanCreationFailure(beanName);

throw ex;

}

}

// Check if required type matches the type of the actual bean instance.

if (requiredType != null && !requiredType.isInstance(bean)) {

try {

T convertedBean = getTypeConverter().convertIfNecessary(bean, requiredType);

if (convertedBean == null) {

throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());

}

return convertedBean;

}

catch (TypeMismatchException ex) {

if (logger.isDebugEnabled()) {

logger.debug(“Failed to convert bean '” + name + “’ to required type '” +

ClassUtils.getQualifiedName(requiredType) + “'”, ex);

}

throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());

}

}

return (T) bean;

}

复制代码

第一步:先尝试去缓存中获取对象

Object sharedInstance = getSingleton(beanName),由于第一次肯定为空:

public Object getSingleton(String beanName) {

//在这里系统一般是允许早期对象引用的allowEarlyReference通过这个参数可以控制解决循环依赖

return getSingleton(beanName, true);

}

进入到getSingleton(beanName, true)方法,这里涉及到Spring的三级缓存,用它来解决循环依赖:

复制代码

protected Object getSingleton(String beanName, boolean allowEarlyReference) {

/**

* 第一步:我们尝试去一级缓存(单例缓存池中去获取对象,一般情况从该map中获取的对象是直接可以使用的)

* Spring IoC容器初始化加载单实例bean的时候第一次进来的时候 该map中一般返回空

*/

Object singletonObject = this.singletonObjects.get(beanName);

/**

* 若在第一级缓存中没有获取到对象,并且singletonsCurrentlyInCreation正在创建的单实例的list包含该beanName

* Spring IoC容器初始化加载单实例bean的时候第一次进来的时候 该list中一般返回空,但是循环依赖的时候可以满足该条件

*/

if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {

synchronized (this.singletonObjects) {

/**

* 尝试去二级缓存中获取对象(二级缓存中的对象是一个早期对象)

* 何为早期对象:就是bean刚刚调用了构造方法,还没给bean的属性进行赋值的对象就是早期对象

*/

singletonObject = this.earlySingletonObjects.get(beanName);

/**

* 二级缓存中也没有获取到对象,allowEarlyReference为true(参数是有上一个方法传递进来的true)

*/

if (singletonObject == null && allowEarlyReference) {

/**

* 直接从三级缓存中获取ObjectFactory对象 这个对接就是用来解决循环依赖的关键所在

* 在getBean的过程中,当bean调用了构造方法的时候,把早期对象包裹成一个ObjectFactory暴露到三级缓存中

*/

ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);

//从三级缓存中获取到对象不为空

if (singletonFactory != null) {

/**

* 在这里通过暴露的ObjectFactory包装对象中,通过调用他的getObject()来获取我们的早期对象

* 在这个环节中会调用到 getEarlyBeanReference()来进行后置处理

*/

singletonObject = singletonFactory.getObject();

//把早期对象放置在二级缓存,

this.earlySingletonObjects.put(beanName, singletonObject);

//ObjectFactory 包装对象从三级缓存中删除掉

this.singletonFactories.remove(beanName);

}

}

}

}

return singletonObject;

}

复制代码

第二步:走else逻辑

Spring只能解决单例对象的setter注入的循环依赖,不能解决构造器注入,也不能解决多实例的循环依赖,所以会抛Bean当前正在创建的异常,接着判断是否有父工厂,有就调用父工厂的getBean,如果不是仅仅做类型检查,而是创建Bean对象,则需要调用markBeanAsCreated方法进行标识

合并Bean定义信息

① 接着调用getMergedLocalBeanDefinition(beanName)进行Bean定义的合并方法如下:

复制代码

protected RootBeanDefinition getMergedLocalBeanDefinition(String beanName) throws BeansException {

// 快速从缓存中获取,如果不为空,则直接返回

RootBeanDefinition mbd = this.mergedBeanDefinitions.get(beanName);

if (mbd != null) {

return mbd;

}

//获取 RootBeanDefinition 对象。若获取的 BeanDefinition 为子 BeanDefinition,则需要合并父类的相关属性.

return getMergedBeanDefinition(beanName, getBeanDefinition(beanName));

}

复制代码

② 进入到getMergedBeanDefinition(beanName, getBeanDefinition(beanName))方法:

protected RootBeanDefinition getMergedBeanDefinition(String beanName, BeanDefinition bd)

throws BeanDefinitionStoreException {

//调用重载的方法

return getMergedBeanDefinition(beanName, bd, null);

}

③ 进入到getMergedBeanDefinition(beanName, bd, null)的方法:

复制代码

protected RootBeanDefinition getMergedBeanDefinition(

String beanName, BeanDefinition bd, @Nullable BeanDefinition containingBd)

throws BeanDefinitionStoreException {

//加锁

synchronized (this.mergedBeanDefinitions) {

RootBeanDefinition mbd = null;

// Check with full lock now in order to enforce the same merged instance.

if (containingBd == null) {

mbd = this.mergedBeanDefinitions.get(beanName);

}

if (mbd == null) {

//bd.getParentName() == null,表明无父配置,这时直接将当前的BeanDefinition升级为RootBeanDefinition

if (bd.getParentName() == null) {

//直接把原始的bean定义升级为RootBeanDefinition

if (bd instanceof RootBeanDefinition) {

mbd = ((RootBeanDefinition) bd).cloneBeanDefinition();

}

else {

//包裹为RootBeanDefinition

mbd = new RootBeanDefinition(bd);

}

}

//有父定义

else {

BeanDefinition pbd;

try {

/*

* 判断父类beanName与子类beanName名称是否相同。若相同,则父类bean一定

* 在父容器中。原因也很简单,容器底层是用Map缓存<beanName, bean> 键值对

* 的。同一个容器下,使用同一个 beanName 映射两个bean实例显然是不合适的

*/

String parentBeanName = transformedBeanName(bd.getParentName());

if (!beanName.equals(parentBeanName)) {

pbd = getMergedBeanDefinition(parentBeanName);

}

else {

/*

* 这里再次调用getMergedBeanDefinition,只不过参数值变为了

* parentBeanName,用于合并父BeanDefinition 和爷爷辈的

* BeanDefinition。如果爷爷辈的BeanDefinition仍有父

* BeanDefinition,则继续合并

*/

BeanFactory parent = getParentBeanFactory();

if (parent instanceof ConfigurableBeanFactory) {

pbd = ((ConfigurableBeanFactory) parent).getMergedBeanDefinition(parentBeanName);

}

else {

throw new NoSuchBeanDefinitionException(parentBeanName,

“Parent name '” + parentBeanName + “’ is equal to bean name '” + beanName +

“': cannot be resolved without an AbstractBeanFactory parent”);

}

}

}

catch (NoSuchBeanDefinitionException ex) {

throw new BeanDefinitionStoreException(bd.getResourceDescription(), beanName,

“Could not resolve parent bean definition '” + bd.getParentName() + “'”, ex);

}

//以父BeanDefinition的配置信息为基本创建RootBeanDefinition

mbd = new RootBeanDefinition(pbd);

//用子BeanDefinition中的属性覆盖父BeanDefinition中的属性

mbd.overrideFrom(bd);

}

// 如果用户未配置scope属性,则默认将该属性配置为singleton

if (!StringUtils.hasLength(mbd.getScope())) {

mbd.setScope(RootBeanDefinition.SCOPE_SINGLETON);

}

// A bean contained in a non-singleton bean cannot be a singleton itself.

// Let’s correct this on the fly here, since this might be the result of

// parent-child merging for the outer bean, in which case the original inner bean

// definition will not have inherited the merged outer bean’s singleton status.

if (containingBd != null && !containingBd.isSingleton() && mbd.isSingleton()) {

mbd.setScope(containingBd.getScope());

}

//缓存合并后的BeanDefinition

if (containingBd == null && isCacheBeanMetadata()) {

this.mergedBeanDefinitions.put(beanName, mbd);

}

}

return mbd;

}

}

复制代码

④ 然后检查当前创建的bean定义是不是抽象的bean定义,checkMergedBeanDefinition(mbd, beanName, args)方法:

复制代码

protected void checkMergedBeanDefinition(RootBeanDefinition mbd, String beanName, @Nullable Object[] args)

throws BeanDefinitionStoreException {

//抽象的bean定义是不能够被实例化的

if (mbd.isAbstract()) {

throw new BeanIsAbstractException(beanName);

}

}

复制代码

处理dependsOn的依赖(这个不是我们所谓的循环依赖 而是bean创建前后的依赖)

① 进入到isDependent(beanName, dependentBeanName, null)的方法:

复制代码

private boolean isDependent(String beanName, String dependentBeanName, @Nullable Set alreadySeen) {

//alreadySeen已经检测的依赖bean

if (alreadySeen != null && alreadySeen.contains(beanName)) {

return false;

}

//获取原始beanName

String canonicalName = canonicalName(beanName);

//获取创建当前bean所依赖的bean的名称集合

Set dependentBeans = this.dependentBeanMap.get(canonicalName);

//不依赖任何前置Bean直接返回

if (dependentBeans == null) {

return false;

}

//存在,则证明存在已经注册的依赖

if (dependentBeans.contains(dependentBeanName)) {

return true;

}

//递归检测依赖

for (String transitiveDependency : dependentBeans) {

if (alreadySeen == null) {

alreadySeen = new HashSet<>();

}

//添加到alreadySeen 中

alreadySeen.add(beanName);

//递归检查依赖

if (isDependent(transitiveDependency, dependentBeanName, alreadySeen)) {

return true;

}

}

return false;

}

复制代码

② 进入到保存的是依赖beanName之间的映射关系:依赖beanName -> beanName的集合registerDependentBean(dep, beanName)的方法:

复制代码

public void registerDependentBean(String beanName, String dependentBeanName) {

//获取原始的beanName

String canonicalName = canonicalName(beanName);

// 添加 <canonicalName, dependentBeanName> 到 dependentBeanMap 中

synchronized (this.dependentBeanMap) {

Set dependentBeans =

this.dependentBeanMap.computeIfAbsent(canonicalName, k -> new LinkedHashSet<>(8));

if (!dependentBeans.add(dependentBeanName)) {

return;

}

}

// 添加 <dependentBeanName, canonicalName> 到 dependenciesForBeanMap 中

synchronized (this.dependenciesForBeanMap) {

Set dependenciesForBean =

this.dependenciesForBeanMap.computeIfAbsent(dependentBeanName, k -> new LinkedHashSet<>(8));

dependenciesForBean.add(canonicalName);

}

}

复制代码

③ 然后到获取dependsOn的Bean的getBean(dep),就获取到了dependsOn的Bean了;

处理单实例Bean

① 把beanName和一个singletonFactory匿名内部类传入用于回调的getSingleton方法:

复制代码

public Object getSingleton(String beanName, ObjectFactory<?> singletonFactory) {

Assert.notNull(beanName, “Bean name must not be null”);

//加锁

synchronized (this.singletonObjects) {

//尝试从单例缓存池中获取对象

Object singletonObject = this.singletonObjects.get(beanName);

if (singletonObject == null) {

if (this.singletonsCurrentlyInDestruction) {

throw new BeanCreationNotAllowedException(beanName,

"Singleton bean creation not allowed while singletons of this factory are in destruction " +

“(Do not request a bean from a BeanFactory in a destroy method implementation!)”);

}

if (logger.isDebugEnabled()) {

logger.debug(“Creating shared instance of singleton bean '” + beanName + “'”);

}

/**

* 标记当前的bean马上就要被创建了

* singletonsCurrentlyInCreation 在这里会把beanName加入进来,若第二次循环依赖(构造器注入会抛出异常)

*/

beforeSingletonCreation(beanName);

boolean newSingleton = false;

boolean recordSuppressedExceptions = (this.suppressedExceptions == null);

if (recordSuppressedExceptions) {

this.suppressedExceptions = new LinkedHashSet<>();

}

try {

//创建bean这个过程其实是调用 createBean() 方法

singletonObject = singletonFactory.getObject();

newSingleton = true;

}

catch (IllegalStateException ex) {

//回调我们singletonObjects的get方法,进行正在的创建bean的逻辑

singletonObject = this.singletonObjects.get(beanName);

if (singletonObject == null) {

throw ex;

}

}

catch (BeanCreationException ex) {

if (recordSuppressedExceptions) {

for (Exception suppressedException : this.suppressedExceptions) {

ex.addRelatedCause(suppressedException);

}

}

throw ex;

}

finally {

if (recordSuppressedExceptions) {

this.suppressedExceptions = null;

}

//后置处理主要做的事情就是把singletonsCurrentlyInCreation标记正在创建的bean从集合中移除

afterSingletonCreation(beanName);

}

if (newSingleton) {

//加入缓存中

addSingleton(beanName, singletonObject);

}

}

return singletonObject;

}

}

复制代码

第一:先到单例缓存池中获取对象获取,有就返回没有就继续往下

第二:调用标记当前的bean马上就要被创建了beforeSingletonCreation(beanName)方法:

复制代码

protected void beforeSingletonCreation(String beanName) {

//若singletonsCurrentlyInCreation没有 则添加成功

if (!this.inCreationCheckExclusions.contains(beanName) && !this.singletonsCurrentlyInCreation.add(beanName)) {

throw new BeanCurrentlyInCreationException(beanName);

}

复制代码

第三:创建Bean的singletonFactory.getObject()其实是调用createBean()方法:

复制代码

protected Object createBean(String beanName, RootBeanDefinition mbd, Object[] args) throws BeanCreationException {

if (logger.isDebugEnabled()) {

logger.debug(“Creating instance of bean '” + beanName + “'”);

}

RootBeanDefinition mbdToUse = mbd;

// Make sure bean class is actually resolved at this point, and

// clone the bean definition in case of a dynamically resolved Class

// which cannot be stored in the shared merged bean definition.

Class<?> resolvedClass = resolveBeanClass(mbd, beanName);

if (resolvedClass != null && !mbd.hasBeanClass() && mbd.getBeanClassName() != null) {

mbdToUse = new RootBeanDefinition(mbd);

mbdToUse.setBeanClass(resolvedClass);

}

// Prepare method overrides.

try {

mbdToUse.prepareMethodOverrides();

}

catch (BeanDefinitionValidationException ex) {

throw new BeanDefinitionStoreException(mbdToUse.getResourceDescription(),

beanName, “Validation of method overrides failed”, ex);

}

try {

// Give BeanPostProcessors a chance to return a proxy instead of the target bean instance.

Object bean = resolveBeforeInstantiation(beanName, mbdToUse);

if (bean != null) {

return bean;

}

}

catch (Throwable ex) {

throw new BeanCreationException(mbdToUse.getResourceDescription(), beanName,

“BeanPostProcessor before instantiation of bean failed”, ex);

}

Object beanInstance = doCreateBean(beanName, mbdToUse, args);

if (logger.isDebugEnabled()) {

logger.debug(“Finished creating instance of bean '” + beanName + “'”);

}

return beanInstance;

}

复制代码

真正的创建我们的bean的实例对象的doCreateBean(beanName, mbdToUse, args)的方法:

复制代码

protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)

throws BeanCreationException {

//BeanWrapper是对Bean的包装,其接口中所定义的功能很简单包括设置获取被包装的对象,获取被包装bean的属性描述器

BeanWrapper instanceWrapper = null;

if (mbd.isSingleton()) {

//从没有完成的FactoryBean中移除

instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);

}

if (instanceWrapper == null) {

//使用合适的实例化策略来创建新的实例:工厂方法、构造函数自动注入、简单初始化 比较复杂也很重要

instanceWrapper = createBeanInstance(beanName, mbd, args);

}

//从beanWrapper中获取我们的早期对象

final Object bean = instanceWrapper.getWrappedInstance();

Class<?> beanType = instanceWrapper.getWrappedClass();

if (beanType != NullBean.class) {

mbd.resolvedTargetType = beanType;

}

// Allow post-processors to modify the merged bean definition.

synchronized (mbd.postProcessingLock) {

if (!mbd.postProcessed) {

try {

//进行后置处理@AutoWired的注解的预解析

applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);

}

catch (Throwable ex) {

throw new BeanCreationException(mbd.getResourceDescription(), beanName,

“Post-processing of merged bean definition failed”, ex);

}

mbd.postProcessed = true;

}

}

/**

* 该对象进行判断是否能够暴露早期对象的条件

* 单实例 this.allowCircularReferences 默认为true

* isSingletonCurrentlyInCreation(表示当前的bean对象正在创建singletonsCurrentlyInCreation包含当前正在创建的bean)

*/

boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&

isSingletonCurrentlyInCreation(beanName));

//上述条件满足,允许中期暴露对象

if (earlySingletonExposure) {

if (logger.isDebugEnabled()) {

logger.debug(“Eagerly caching bean '” + beanName +

“’ to allow for resolving potential circular references”);

}

//把我们的早期对象包装成一个singletonFactory对象 该对象提供了一个getObject方法,该方法内部调用getEarlyBeanReference方法

addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));

}

// Initialize the bean instance.

Object exposedObject = bean;

try {

//给我们的属性进行赋值(调用set方法进行赋值)

populateBean(beanName, mbd, instanceWrapper);

//进行对象初始化操作(在这里可能生成代理对象)

exposedObject = initializeBean(beanName, exposedObject, mbd);

}

catch (Throwable ex) {

if (ex instanceof BeanCreationException && beanName.equals(((BeanCreationException) ex).getBeanName())) {

throw (BeanCreationException) ex;

}

else {

throw new BeanCreationException(

mbd.getResourceDescription(), beanName, “Initialization of bean failed”, ex);

}

}

//允许早期对象的引用

if (earlySingletonExposure) {

/**

* 去缓存中获取到我们的对象 由于传递的allowEarlyReference 是false 要求只能在一级二级缓存中去获取

* 正常普通的bean(不存在循环依赖的bean) 创建的过程中,压根不会把三级缓存提升到二级缓存中

*/

Object earlySingletonReference = getSingleton(beanName, false);

//能够获取到

if (earlySingletonReference != null) {

//经过后置处理的bean和早期的bean引用还相等的话(表示当前的bean没有被代理过)

if (exposedObject == bean) {

exposedObject = earlySingletonReference;

}

//处理依赖的bean

else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {

String[] dependentBeans = getDependentBeans(beanName);

Set actualDependentBeans = new LinkedHashSet<>(dependentBeans.length);

for (String dependentBean : dependentBeans) {

if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {

actualDependentBeans.add(dependentBean);

}

}

if (!actualDependentBeans.isEmpty()) {

throw new BeanCurrentlyInCreationException(beanName,

“Bean with name '” + beanName + “’ has been injected into other beans [” +

StringUtils.collectionToCommaDelimitedString(actualDependentBeans) +

"] in its raw version as part of a circular reference, but has eventually been " +

"wrapped. This means that said other beans do not use the final version of the " +

"bean. This is often the result of over-eager type matching - consider using " +

“‘getBeanNamesOfType’ with the ‘allowEagerInit’ flag turned off, for example.”);

}

}

}

}

// Register bean as disposable.

try {

//注册销毁的bean的销毁接口

registerDisposableBeanIfNecessary(beanName, bean, mbd);

}

catch (BeanDefinitionValidationException ex) {

throw new BeanCreationException(

mbd.getResourceDescription(), beanName, “Invalid destruction signature”, ex);

}

return exposedObject;

}

复制代码

① 实例化:使用合适的实例化策略来创建新的实例:工厂方法、构造函数自动注入、简单初始化的createBeanInstance方法:

复制代码

protected BeanWrapper createBeanInstance(String beanName, RootBeanDefinition mbd, @Nullable Object[] args) {

//从bean定义中解析出当前bean的class对象

Class<?> beanClass = resolveBeanClass(mbd, beanName);

//检测类的访问权限。默认情况下,对于非 public 的类,是允许访问的。若禁止访问,这里会抛出异常

if (beanClass != null && !Modifier.isPublic(beanClass.getModifiers()) && !mbd.isNonPublicAccessAllowed()) {

throw new BeanCreationException(mbd.getResourceDescription(), beanName,

"Bean class isn’t public, and non-public access not allowed: " + beanClass.getName());

}

Supplier<?> instanceSupplier = mbd.getInstanceSupplier();

if (instanceSupplier != null) {

return obtainFromSupplier(instanceSupplier, beanName);

}

//工厂方法,我们通过配置类来进行配置的话 采用的就是工厂方法

if (mbd.getFactoryMethodName() != null) {

return instantiateUsingFactoryMethod(beanName, mbd, args);

}

//判断当前构造函数是否被解析过

boolean resolved = false;

//有没有必须进行依赖注入

boolean autowireNecessary = false;

/**

* 通过getBean传入进来的构造函数是否来指定需要推断构造函数

* 若传递进来的args不为空,那么就可以直接选出对应的构造函数

*/

if (args == null) {

synchronized (mbd.constructorArgumentLock) {

//判断我们的bean定义信息中的resolvedConstructorOrFactoryMethod(用来缓存我们的已经解析的构造函数或者工厂方法)

if (mbd.resolvedConstructorOrFactoryMethod != null) {

//修改已经解析过的构造函数的标志

resolved = true;

//修改标记为ture 标识构造函数或者工厂方法已经解析过

autowireNecessary = mbd.constructorArgumentsResolved;

}

}

}

//若被解析过

if (resolved) {

if (autowireNecessary) {

//通过有参的构造函数进行反射调用

return autowireConstructor(beanName, mbd, null, null);

}

else {

//调用无参数的构造函数进行创建对象

return instantiateBean(beanName, mbd);

}

}

//通过bean的后置处理器进行选举出合适的构造函数对象

Constructor<?>[] ctors = determineConstructorsFromBeanPostProcessors(beanClass, beanName);

//通过后置处理器解析出构造器对象不为null或获取bean定义中的注入模式是构造器注入或bean定义信息ConstructorArgumentValues或获取通过getBean的方式传入的构造器函数参数类型不为null

if (ctors != null || mbd.getResolvedAutowireMode() == AUTOWIRE_CONSTRUCTOR ||

mbd.hasConstructorArgumentValues() || !ObjectUtils.isEmpty(args)) {

//通过构造函数创建对象

return autowireConstructor(beanName, mbd, ctors, args);

}

//使用无参数的构造函数调用创建对象

return instantiateBean(beanName, mbd);

}

复制代码

判断是否暴露早期对象条件满足就暴露早期对象,把我们的早期对象包装成一个singletonFactory对象 该对象提供了一个getObject方法,该方法内部调用getEarlyBeanReference方法:

复制代码

protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) {

Object exposedObject = bean;

//判读我们容器中是否有InstantiationAwareBeanPostProcessors类型的后置处理器

if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {

//获取我们所有的后置处理器

for (BeanPostProcessor bp : getBeanPostProcessors()) {

//判断我们的后置处理器是不是实现了SmartInstantiationAwareBeanPostProcessor接口

if (bp instanceof SmartInstantiationAwareBeanPostProcessor) {

//进行强制转换

SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp;

//挨个调用SmartInstantiationAwareBeanPostProcessor的getEarlyBeanReference

exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName);

}

}

}

return exposedObject;

}

复制代码

addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean))方法:

复制代码

/**

* 该方法用于把早期对象包装成一个ObjectFactory 暴露到三级缓存中 用于将解决循环依赖

* @param beanName the name of the bean

* @param singletonFactory the factory for the singleton object

*/

protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {

Assert.notNull(singletonFactory, “Singleton factory must not be null”);

//加锁

synchronized (this.singletonObjects) {

//单例缓存池中没有包含当前的bean

if (!this.singletonObjects.containsKey(beanName)) {

//加入到三级缓存中 暴露早期对象用于解决循环依赖

this.singletonFactories.put(beanName, singletonFactory);

this.earlySingletonObjects.remove(beanName);

this.registeredSingletons.add(beanName);

}

}

}

复制代码

②属性赋值:给我们的属性进行赋值(调用set方法进行赋值)populateBean(beanName, mbd, instanceWrapper)方法:

复制代码

/**

*给我们的对象BeanWrapper属性赋值

* @param beanName bean的名称

* @param mbd bean的定义

* @param bw bean实例包装对象

*/

protected void populateBean(String beanName, RootBeanDefinition mbd, @Nullable BeanWrapper bw) {

//若bw为null的话,则说明对象没有实例化

if (bw == null) {

//进入if说明对象有属性,bw为空,不能为他设置属性,那就在下面就执行抛出异常

if (mbd.hasPropertyValues()) {

throw new BeanCreationException(

mbd.getResourceDescription(), beanName, “Cannot apply property values to null instance”);

}

else {

// Skip property population phase for null instance.

return;

}

}

/**

* 在属性被填充前,给 InstantiationAwareBeanPostProcessor 类型的后置处理器一个修改

* bean 状态的机会。官方的解释是:让用户可以自定义属性注入。比如用户实现一

* 个 InstantiationAwareBeanPostProcessor 类型的后置处理器,并通过

* postProcessAfterInstantiation 方法向 bean 的成员变量注入自定义的信息。

*当时我们发现系统中的的InstantiationAwareBeanPostProcessor.postProcessAfterInstantiationM没有进行任何处理,

*若我们自己实现了这个接口 可以自定义处理…spring 留给我们自己扩展接口的

*特殊需求,直接使用配置中的信息注入即可。

*/

boolean continueWithPropertyPopulation = true;

//是否持有 InstantiationAwareBeanPostProcessor

if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {

//获取容器中的所有的BeanPostProcessor

for (BeanPostProcessor bp : getBeanPostProcessors()) {

//判断我们的后置处理器是不是InstantiationAwareBeanPostProcessor

if (bp instanceof InstantiationAwareBeanPostProcessor) {

//进行强制转化

InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;

//若存在后置处理器给我们属性赋值了,那么返回false 可以来修改我们的开关变量,就不会走下面的逻辑了

if (!ibp.postProcessAfterInstantiation(bw.getWrappedInstance(), beanName)) {

// 返回值为是否继续填充 bean

// postProcessAfterInstantiation:如果应该在 bean上面设置属性则返回 true,否则返回 false

// 一般情况下,应该是返回true 。

// 返回 false 的话,将会阻止在此 Bean 实例上调用任何后续的 InstantiationAwareBeanPostProcessor 实

continueWithPropertyPopulation = false;

break;

完结

Redis基于内存,常用作于缓存的一种技术,并且Redis存储的方式是以key-value的形式。Redis是如今互联网技术架构中,使用最广泛的缓存,在工作中常常会使用到。Redis也是中高级后端工程师技术面试中,面试官最喜欢问的问题之一,因此作为Java开发者,Redis是我们必须要掌握的。

Redis 是 NoSQL 数据库领域的佼佼者,如果你需要了解 Redis 是如何实现高并发、海量数据存储的,那么这份腾讯专家手敲《Redis源码日志笔记》将会是你的最佳选择。

rty values to null instance");

}

else {

// Skip property population phase for null instance.

return;

}

}

/**

* 在属性被填充前,给 InstantiationAwareBeanPostProcessor 类型的后置处理器一个修改

* bean 状态的机会。官方的解释是:让用户可以自定义属性注入。比如用户实现一

* 个 InstantiationAwareBeanPostProcessor 类型的后置处理器,并通过

* postProcessAfterInstantiation 方法向 bean 的成员变量注入自定义的信息。

*当时我们发现系统中的的InstantiationAwareBeanPostProcessor.postProcessAfterInstantiationM没有进行任何处理,

*若我们自己实现了这个接口 可以自定义处理…spring 留给我们自己扩展接口的

*特殊需求,直接使用配置中的信息注入即可。

*/

boolean continueWithPropertyPopulation = true;

//是否持有 InstantiationAwareBeanPostProcessor

if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {

//获取容器中的所有的BeanPostProcessor

for (BeanPostProcessor bp : getBeanPostProcessors()) {

//判断我们的后置处理器是不是InstantiationAwareBeanPostProcessor

if (bp instanceof InstantiationAwareBeanPostProcessor) {

//进行强制转化

InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;

//若存在后置处理器给我们属性赋值了,那么返回false 可以来修改我们的开关变量,就不会走下面的逻辑了

if (!ibp.postProcessAfterInstantiation(bw.getWrappedInstance(), beanName)) {

// 返回值为是否继续填充 bean

// postProcessAfterInstantiation:如果应该在 bean上面设置属性则返回 true,否则返回 false

// 一般情况下,应该是返回true 。

// 返回 false 的话,将会阻止在此 Bean 实例上调用任何后续的 InstantiationAwareBeanPostProcessor 实

continueWithPropertyPopulation = false;

break;

完结

Redis基于内存,常用作于缓存的一种技术,并且Redis存储的方式是以key-value的形式。Redis是如今互联网技术架构中,使用最广泛的缓存,在工作中常常会使用到。Redis也是中高级后端工程师技术面试中,面试官最喜欢问的问题之一,因此作为Java开发者,Redis是我们必须要掌握的。

Redis 是 NoSQL 数据库领域的佼佼者,如果你需要了解 Redis 是如何实现高并发、海量数据存储的,那么这份腾讯专家手敲《Redis源码日志笔记》将会是你的最佳选择。

[外链图片转存中…(img-ONQbaAA9-1714456835068)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

  • 15
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值