bean 的加载(四)
前一篇文章次要解说了对于 Spring 循环依赖相干的问题。本文持续解说对于 bean 的加载过程之创立 bean。
创立 Bean
咱们先进入到 createBean 办法中。
之前文章咱们说过在经验了 resolveBeforeInstantiation 办法后,程序会有两个抉择,如果创立了代理或者重写了 InstantiationAwareBeanPostProcessor 中的 postProcessBeforeInstantiation 办法并在办法 postProcessBeforeInstantiation 中扭转了 bean,则会间接返回,否则会进行惯例 bean 的创立。
// Give BeanPostProcessors a chance to return a proxy instead of the target bean instance.
//给BeanPostProcessors一个机会来返回代理用于代替真正的实例
Object bean = resolveBeforeInstantiation(beanName, mbdToUse);
if (bean != null) {
return bean;
}
惯例 bean 的创立则在 doCreateBean 中。
//代码(1)
Object beanInstance = doCreateBean(beanName, mbdToUse, args);
protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)
throws BeanCreationException {
// Instantiate the bean.
BeanWrapper instanceWrapper = null;
//如果为单例则移除掉缓存
if (mbd.isSingleton()) {
instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);
}
if (instanceWrapper == null) {
//依据指定bean应用的对应策略创立新的实例,比方结构器注入,工厂办法,简略初始化始化
instanceWrapper = createBeanInstance(beanName, mbd, args);
}
final Object bean = instanceWrapper.getWrappedInstance();
Class<?> beanType = instanceWrapper.getWrappedClass();
if (beanType != NullBean.class) {
// 将解析类型设置为beanType
mbd.resolvedTargetType = beanType;
}
// Allow post-processors to modify the merged bean definition.
synchronized (mbd.postProcessingLock) {
if (!mbd.postProcessed) {
try {
// 这里次要是 MergedBeanDefinitionPostProcessor 对@Autowire,@Value这些注解进行解决
applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
} catch (Throwable ex) {
throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Post-processing of merged bean definition failed", ex);
}
mbd.postProcessed = true;
}
}
// Eagerly cache singletons to be able to resolve circular references
// even when triggered by lifecycle interfaces like BeanFactoryAware.
//是否须要提前曝光,单例&&容许循环依赖&&以后bean正在创立中,检测循环依赖
boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
if (logger.isTraceEnabled()) {
logger.trace("Eagerly caching bean '" + beanName +
"' to allow for resolving potential circular references");
}
// 为防止前期循环依赖,能够在 bean 初始化实现前将创立实例的 objectFactory退出缓存
// 对bean 再一次依赖援用,次要利用SmartInstantiationAwareBeanPostProcessor
// 其中咱们相熟的AOP就是在这里将advice 动静织入,若没有间接返回bean
addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}
// Initialize the bean instance.
Object exposedObject = bean;
try {
//对bean进行填充,比方设置属性,其中可能存在依赖其余bean,则会递归初始化依赖的bean
populateBean(beanName, mbd, instanceWrapper);
// 进一步初始化Bean
// 注入 Aware 相干的对象
// 调用 后置处理器 BeanPostProcessor 外面的postProcessBeforeInitialization办法
// 调用 InitializingBean中的的 afterPropertiesSet()
// 调用 init-method,调用相应的init办法
// 调用 后置处理器 BeanPostProcessor 外面的调用实现的postProcessAfterInitialization办法
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) {
Object earlySingletonReference = getSingleton(beanName, false);
// earlySingletonReference 只有在检测到有循环依赖的状况下才会不为空
if (earlySingletonReference != null) {
//如果exposedObject 没有在初始化办法中被扭转,也就是没有被加强
if (exposedObject == bean) {
exposedObject = earlySingletonReference;
} else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {
String[] dependentBeans = getDependentBeans(beanName);
Set<String> actualDependentBeans = new LinkedHashSet<>(dependentBeans.length);
// 查看依赖
for (String dependentBean : dependentBeans) {
if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {
actualDependentBeans.add(dependentBean);
}
}
/**
因为 bean 创立后其所依赖的bean肯定是曾经创立,
actualDependentBeans 不为空则示意 以后bean 创立后其依赖的bean 却没有全副创立,
也就是说存在依赖
*/
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 {
// 注册到 disposableBeans 外面,以便在销毁bean 的时候 能够运行指定的相干业务
registerDisposableBeanIfNecessary(beanName, bean, mbd);
} catch (BeanDefinitionValidationException ex) {
throw new BeanCreationException(
mbd.getResourceDescription(), beanName, "Invalid destruction signature", ex);
}
return exposedObject;
}
这个办法比拟长,咱们先梳理一下思路。
- 如果是单例则须要首先革除缓存
-
实例化 bean,将 BeanDefinition 转换为 BeanWrapper
转换是一个比较复杂的过程,大抵如下:
- 如果存在工厂办法则应用工厂办法进行初始化
- 一个类有多个构造函数,所以须要依据参数锁定构造函数并初始化
- 如果不存在工厂办法也不存在带参数的构造函数,则应用默认的无参构造函数进行初始化
- mergedBeanDefinitionPostProcessor 的利用。bean 合并后的解决,Autowired 注解就是通过此办法实现的预解析
- 依赖解决。解决循环依赖,上一章曾经讲过,就是将须要注入的实例放入缓存中的 ObjectFactory 来解决
- 属性填充。将所有属性填充到 bean 的实例中
- 循环依赖查看。之前说过,Spring 中只能解决单例的循环依赖,对于别的 scope 属性的 bean,Spring 在这个步骤会检测是否存在循环依赖,如果存在则抛出异样
- 注册 DisposableBean。如果配置了 destory-method,这里须要注册以便在销毁时调用
- 实现创立并返回
能够看到步骤十分繁琐,咱们接下来一步步进行理解。
创立 bean 的实例
咱们先从 createBeanInstance 办法开始理解。
protected BeanWrapper createBeanInstance(String beanName, RootBeanDefinition mbd, @Nullable Object[] args) {
// Make sure bean class is actually resolved at this point.
//确保bean曾经被解析
Class<?> beanClass = resolveBeanClass(mbd, beanName);
// 如果beanClass 不是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());
}
// 如果存在,就返回一个 callback回调函数,在 obtainFromSupplier 办法里调用对应的具体方法 ,并转换成 BeanWrapper 类型
Supplier<?> instanceSupplier = mbd.getInstanceSupplier();
if (instanceSupplier != null) {
return obtainFromSupplier(instanceSupplier, beanName);
}
//如果工厂办法不为空则应用工厂办法初始化策略
if (mbd.getFactoryMethodName() != null) {
return instantiateUsingFactoryMethod(beanName, mbd, args);
}
// Shortcut when re-creating the same bean...
boolean resolved = false;
boolean autowireNecessary = false;
if (args == null) {
synchronized (mbd.constructorArgumentLock) {
//一个类有多个构造函数,每个构造函数都有不同的参数,所以调用前须要先依据参数锁定构造函数或对应的工厂办法
if (mbd.resolvedConstructorOrFactoryMethod != null) {
resolved = true;
autowireNecessary = mbd.constructorArgumentsResolved;
}
}
}
//如果曾经解析过则应用解析好的构造函数,不须要再次指定
if (resolved) {
if (autowireNecessary) {
//构造函数主动注入
return autowireConstructor(beanName, mbd, null, null);
} else {
//应用默认的构造函数
return instantiateBean(beanName, mbd);
}
}
// Candidate constructors for autowiring?
//须要依据参数解析构造函数
Constructor<?>[] ctors = determineConstructorsFromBeanPostProcessors(beanClass, beanName);
if (ctors != null || mbd.getResolvedAutowireMode() == AUTOWIRE_CONSTRUCTOR ||
mbd.hasConstructorArgumentValues() || !ObjectUtils.isEmpty(args)) {
//构造函数主动注入
return autowireConstructor(beanName, mbd, ctors, args);
}
// Preferred constructors for default construction?
// 如果下面没有解析好对应的构造函数, 这里看看有没有指定构造函数
/**
具体外面其实 是 SmartInstantiationAwareBeanPostProcessor , 这个类 继承了
InstantiationAwareBeanPostProcessor, 调用外面的determineCandidateConstructors办法来确认有没有指定的构造函数
*/
ctors = mbd.getPreferredConstructors();
if (ctors != null) {
return autowireConstructor(beanName, mbd, ctors, null);
}
// No special handling: simply use no-arg constructor.
//应用默认的构造函数
return instantiateBean(beanName, mbd);
}
尽管代码中实例化的细节很简单,然而咱们能够梳理进去以下逻辑:
- 如果在 RootBeanDefinition 中存在 factoryMethodName 属性,或者在配置文件中配置了 factory-method 那么就会调用
instantiateUsingFactoryMethod(beanName, mbd, args)
办法,依据 RootBeanDefinition 中的配置生成 bean 的实例。 - 解析构造函数并进行构造函数初始化,Spring 依据参数以及类型去判断最终应用哪一个构造函数进行实例化。因为判断比拟耗费性能,所以采纳缓存机制,如果曾经解析过则不须要反复解析,而是从 RootBeanDefinition 中的属性 resolvedConstructorOrFactoryMethod 缓存的值去取,否则须要进行解析,并将解析后果增加到 resolvedConstructorOrFactoryMethod 中。
解析autowireConstructor
办法
/**
* beanName: Bean的名称
* mbd: 该bean的BeanDefinition
* chosenCtors: 该类的结构器数组
* explicitArgs:这个办法时通过getBean办法过去的,那么既然是getBea过去的,那么在咱们getBean的时候除了传入beanName/beanClass以外,还能够传入其余参数,如果传入了其余参数,那么Spring
* 则认为这些参数是结构器初始化对象时的构造方法参数列表,而这个其余参数就是此时的explicitArgs.
*/
public BeanWrapper autowireConstructor(String beanName, RootBeanDefinition mbd,
@Nullable Constructor<?>[] chosenCtors, @Nullable Object[] explicitArgs) {
BeanWrapperImpl bw = new BeanWrapperImpl();
/**
* 初始化BeanWrapper
* */
this.beanFactory.initBeanWrapper(bw);
/**
* constructorToUse是咱们实际上应用的结构器
* */
Constructor<?> constructorToUse = null;
/**
* argsHolderToUse用来存储用到的结构器的参数,上面的argsToUse的值也是从这个argsHolderToUse中取出来的
*/
ArgumentsHolder argsHolderToUse = null;
/**
* 构造方法中应用到的参数列表理论的值
* */
Object[] argsToUse = null;
/**
* 总结一下这个if和else:
* 1、判断是否传入结构参数值列表,如果传入则赋值
* 2、没有传入则从缓存中去取
* */
/**
* 如果咱们getBean的时候传入了参数,那么Spring就认为咱们心愿依照指定的结构参数列表去寻找结构器并实例化对象.
* 那么这里如果不为空则实际上须要应用的构造方法参数列表值就曾经确定了
* */
if (explicitArgs != null) {
argsToUse = explicitArgs;
}
else {
/**
* 这里干的事件很简略,如果这个bean是原型的,那么阐明此办法必定进入过,那么也必定找到过适合的构造方法和结构参数值列表,在找到适合的构造方法和结构参数值列表后会退出到缓存外面去,那么此处如果不是第一次进入的话,那么缓存外面曾经有了不必再次去获取.
* 此处做的工作就是从缓存中去获取曾经找到过并存进来的构造方法和结构参数值列表.
* 还有留神,只有当咱们参数中explicitArgs为空的时候,结构器才会被缓存
* */
Object[] argsToResolve = null;
synchronized (mbd.constructorArgumentLock) {
constructorToUse = (Constructor<?>) mbd.resolvedConstructorOrFactoryMethod;
if (constructorToUse != null && mbd.constructorArgumentsResolved) {
// Found a cached constructor...
argsToUse = mbd.resolvedConstructorArguments;
if (argsToUse == null) {
argsToResolve = mbd.preparedConstructorArguments;
}
}
}
// 如果缓存中存在则赋值.
if (argsToResolve != null) {
argsToUse = resolvePreparedArguments(beanName, mbd, bw, constructorToUse, argsToResolve);
}
}
/**
* 这里判断constructorToUse是否为空是因为下面有可能从缓存外面曾经拿到了,如果拿到了则不须要进if外面去寻找,间接去调用创立实例化操作了.
* 这里重要的是if外面的代码.
* */
if (constructorToUse == null) {
// Need to resolve the constructor.
boolean autowiring = (chosenCtors != null ||
mbd.getResolvedAutowireMode() == AutowireCapableBeanFactory.AUTOWIRE_CONSTRUCTOR);
/**
* 结构器应用的参数
* */
ConstructorArgumentValues resolvedValues = null;
/**
* 最小参数个数,此值须要用来在循环寻找结构器时应用.
* 如果当以后循环到的结构器参数值个数小于这个最小值的话,那么阐明就是不适合的,没必要继续下去.
* */
int minNrOfArgs;
/**
* 如果咱们getBean的中央传入结构参数值列表,那么则最小参数个数就是咱们传入的列表长度
* */
if (explicitArgs != null) {
minNrOfArgs = explicitArgs.length;
}
else {
/**
* 如果咱们没有传入结构器参数值列表,那么则去解析看有没有配置结构器参数列表,例如如下配置:
* <bean class="com.dh.aop.package1.Demo2" id="demo2">
* <constructor-arg index="0" value="111"></constructor-arg>
* <constructor-arg index="1" value="222"></constructor-arg>
* </bean>
* 这个时候,minNrOfArgs的值就是2
* 如果咱们没有配置结构器参数的话,这个minNrOfArgs的值就是0
* */
ConstructorArgumentValues cargs = mbd.getConstructorArgumentValues();
resolvedValues = new ConstructorArgumentValues();
minNrOfArgs = resolveConstructorArguments(beanName, mbd, bw, cargs, resolvedValues);
}
// Take specified constructors, if any.
/**
* 将该办法的参数结构器列表赋值给candidates
* */
Constructor<?>[] candidates = chosenCtors;
/**
* 如果传入的结构器列表为空,则通过class对象去拿
* 如果bd中设置了容许拜访非public的结构器,那么则获取所有的结构器,否则获取public的结构器.
* 留神这里isNonPublicAccessAllowed的默认值为true.
* 如果获取结构器的时候出错当然就要抛异样.
* */
if (candidates == null) {
Class<?> beanClass = mbd.getBeanClass();
try {
candidates = (mbd.isNonPublicAccessAllowed() ?
beanClass.getDeclaredConstructors() : beanClass.getConstructors());
}
catch (Throwable ex) {
throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Resolution of declared constructors on bean Class [" + beanClass.getName() +
"] from ClassLoader [" + beanClass.getClassLoader() + "] failed", ex);
}
}
/**
* 对结构器进行排序:
* public的大于其余的权限
* 如果都是public的,那么参数越多越靠前.
* 能够看这个sort办法外面能够看到的
* */
AutowireUtils.sortConstructors(candidates);
/**
* 差别变量
* */
int minTypeDiffWeight = Integer.MAX_VALUE;
/**
* 有歧义的结构器,就是参数数量统一的,这种状况下的结构器就被列为有歧义的.
* 失常状况下,如果呈现有歧义的结构器,那么就应用第一个,这取决于spring设置的宽松模式.
* 默认为宽松,如此的话就默认应用第一个结构器应用
* 如果设置为严格,则会报错
* 设置宽松/严格模式标记:beanDefinition.setLenientConstructorResolution
* */
Set<Constructor<?>> ambiguousConstructors = null;
LinkedList<UnsatisfiedDependencyException> causes = null;
/**
* 上面就是循环的拿结构器去校验判断选取一个适合的结构器了.在此之前咱们总结一下上述代码做的事件.
* 1、定义constructorToUse、argsHolderToUse、argsToUse,这些别离用来存前面实际上须要应用的结构器、结构器参数、值等
* 2、如果getBean调用的时候传入了结构器参数,那么argsToUse的值就被赋值为传入的结构器参数,否则就尝试从缓存外面去拿constructorToUse和argsToUse
* 这个缓存就是当bean不是原型的时候实例化时找到的适合的结构器等参数,当然如果是第一次进来,或者bean是单例的,那么此缓存中必定没有这个bean相干的结构器数据
* 3、如果缓存外面有,则间接实例化bean后放到wrapper中并return,如果不存在则须要再次进行一些操作
* 4、在不存在时,首先定义resolvedValues,这个是后续循环外面须要应用到的结构器应用的参数列表,定义minNrOfArgs,这个是最小参数个数,首先如果getBean传入了结构器参数
* 那么此值就是传入结构参数的长度,否则就尝试看咱们有没有配置应用某个结构器,如果都没有,那么这个值就是0了,这个变量用来前面在循环结构器的时候筛选用的
* 5、而后定义candidates变量,而后将chosenCtors(是后面传入的结构器列表)赋值过来,如果它为空,那么则须要去通过class去拿结构器,拿的时候判断了一手
* 去拿BeanDefinition中的isNonPublicAccessAllowed,这个isNonPublicAccessAllowed意思为是否容许拜访非public的结构器,如果为true,则去获取所有的结构器,否则只获取public的
* 6、而后对所有的结构器进行排序,规定为public>其余权限,参数个数多的>参数个数少的,至于为什么排序这个可能是spring认为参数越多的越迷信吧
* 7、差别变量,这个看循环外面的代码能力了解
* 8、定义ambiguousConstructors为有歧义的结构器,意思就是如果两个结构器参数统一,那Spring就不晓得该去用哪个,这时这两个结构器就被放入ambiguousConstructors汇合中,他们两个就是有歧义的结构器
* ================================================================================
* 上面循环外面须要搞清楚的就是它具体是如何选取到适合的结构器来应用
* */
for (Constructor<?> candidate : candidates) {
/**
* 拿到以后构造方法的参数class数组
* */
Class<?>[] paramTypes = candidate.getParameterTypes();
/**
* 后面说了[constructorToUse]这个变量是以后确定应用的结构器,如果它不为空,那么阐明咱们曾经确定了应用哪个结构器,那么就没必要继续下去了.
* 但[argsToUse.length > paramTypes.length]这个就比拟难了解.
* 留神每次循环当前argsToUse的值就会扭转为那次循环的结构器的参数,如果以后拿到的argsToUse参数列表的长度大于以后这个结构器的长度,那么阐明上一次拿到的这个argsToUse比以后的这个更适合(下面也说过,Spring认为参数越多的越迷信)
* 这里能够留神一下后面sort排序的时候,结构参数个数越多的越靠前,所以这里敢于用长度判断后间接break,因为如果上一次循环的结构器参数列表为2个,那么这一次(也就是下一次)的结构参数列表必定不会比2大,那么阐明对于参数个数而言,上一次的参数个数必定不会比这一次少,那么必定就更适合了呗
*/
if (constructorToUse != null && argsToUse.length > paramTypes.length) {
// Already found greedy constructor that can be satisfied ->
// do not look any further, there are only less greedy constructors left.
break;
}
/**
* 如果以后结构器的参数数量比最小参数列表数量小的时候,那么跳过这个结构器.
* minNrOfArgs的值有两个中央赋值了:
* 1、如果咱们getBean时传入了其余参数,那么其余参数的个数就是minNrOfArgs的值
* 2、如果咱们getBean没有传参数,那么minNrOfArgs的值就是咱们配置让Spring指定应用某些参数的结构器,那么咱们配置的参数列表数量也就是以后的minNrOfArgs
* 3、如果上述的状况都不存在,那么minNrOfArgs就是0了,大多数时候都是这种状况,如果都没配置,那么就得Spring本人缓缓而不存在此处的筛选了.
* 所以总结来说此处就是做了一个依据咱们本人定义的来筛选的操作
* */
if (paramTypes.length < minNrOfArgs) {
continue;
}
/**
* 存储结构器须要的参数
* */
ArgumentsHolder argsHolder;
/**
* 此处resolvedValues不为空则阐明getBean时传入的参数explicitArgs为空的,
* 因为下面的代码是如果explicitArgs不为空则不对resolvedValues赋值,否则就对resolvedValues赋值
* 此处先看else的代码,会更清晰.
* 如果传入的参数为空,那么则会去拿参数了
* */
if (resolvedValues != null) {
try {
/**
* 去拿到参数列表的名称,这里ConstructorPropertiesChecker.evaluate是这样做的:
* 如果构造方法上退出了ConstructorProperties注解,那么阐明咱们参数名称数组,如果没有这个注解,那么次数paramNames为空的
* */
String[] paramNames = ConstructorPropertiesChecker.evaluate(candidate, paramTypes.length);
/** 这里为空则代表咱们没有通过注解去自定义参数名称,则通过ParameterNameDiscoverer去解析拿到结构器的参数名称列表 */
if (paramNames == null) {
ParameterNameDiscoverer pnd = this.beanFactory.getParameterNameDiscoverer();
if (pnd != null) {
/** 解析拿到参数名称列表 */
paramNames = pnd.getParameterNames(candidate);
}
}
/**
* 此处会去获取这些参数名称的参数值,如果是主动注入的就会通过getBean获取,以后这种结构器注入的状况如果循环依赖则会报错的.
* 这里咱们只须要晓得,此处将结构器须要的参数值拿进去后并封装到了argsHolder中去.
* 当然如果你结构器外面给个Integer的参数,那必定是会报错的,因为这外面会去Spring容器中拿这个Integer,后果呢,必定是NoSuchBeanDefinitionException了
* 其余这里不必太过于细究,有趣味能够具体看.
* */
argsHolder = createArgumentArray(beanName, mbd, resolvedValues, bw, paramTypes, paramNames,
getUserDeclaredConstructor(candidate), autowiring);
}
catch (UnsatisfiedDependencyException ex) {
if (logger.isTraceEnabled()) {
logger.trace("Ignoring constructor [" + candidate + "] of bean '" + beanName + "': " + ex);
}
// Swallow and try next constructor.
if (causes == null) {
causes = new LinkedList<>();
}
causes.add(ex);
continue;
}
}
else {
/**
* 到了这个else外面来,阐明getBean调用的时候传入了结构器参数,那么就阐明咱们心愿依照指定的结构器去初始化Bean.
* 那么这里就须要判断以后结构器的参数个数是否和咱们心愿的个数一样,如果不是,那么就循环去找下一个结构器,
* 如果和咱们心愿的是一样的,那么就将咱们给的参数封装到argsHolder外面去
* */
// Explicit arguments given -> arguments length must match exactly.
if (paramTypes.length != explicitArgs.length) {
continue;
}
argsHolder = new ArgumentsHolder(explicitArgs);
}
/**
* 当达到这里的时候,至此咱们拿到了:
* 1、结构器
* 2、结构器须要的参数和值
* 那么这里就去结算后面定义的那个差别值.
* 留神这里的:isLenientConstructorResolution意思是是否为宽松的模式,为true的时候是宽松,false的时候是严格,默认为true,这个货色后面曾经说了.
* 这个差别值越小越那就阐明越适合.
* 具体差别值如何计算出来的这个能够自行去看外面的代码,argsHolder.getTypeDifferenceWeight(paramTypes)
* */
int typeDiffWeight = (mbd.isLenientConstructorResolution() ?
argsHolder.getTypeDifferenceWeight(paramTypes) : argsHolder.getAssignabilityWeight(paramTypes));
// Choose this constructor if it represents the closest match.
/**
* 如果本次计算到的差别值比上一次获取到的差别值小,那么就须要做这几件事:
* 1、设置constructorToUse为以后的这个结构器
* 2、设置参数和参数值
* 3、给差别值赋值为以后计算出来的差别值
* 4、清空有歧义的汇合(因为此时咱们曾经失去了更适合的结构器,所以有歧义的结构器外面保留的结构器曾经没有存在的意义了)
* */
if (typeDiffWeight < minTypeDiffWeight) {
constructorToUse = candidate;
argsHolderToUse = argsHolder;
argsToUse = argsHolder.arguments;
minTypeDiffWeight = typeDiffWeight;
ambiguousConstructors = null;
}
/**
* 如果曾经找到过一次结构器,并且以后的差别值和上一次的差别值统一的话,那么阐明这两个结构器是有歧义的,
* 那么就将这两个结构器放到[有歧义的结构器]汇合中去.
* */
else if (constructorToUse != null && typeDiffWeight == minTypeDiffWeight) {
if (ambiguousConstructors == null) {
ambiguousConstructors = new LinkedHashSet<>();
ambiguousConstructors.add(constructorToUse);
}
ambiguousConstructors.add(candidate);
}
}
/**
* 达到这里的时候,如果还没有找到适合的结构器,那么则间接抛出异样,这个类没法实例化
* */
if (constructorToUse == null) {
if (causes != null) {
UnsatisfiedDependencyException ex = causes.removeLast();
for (Exception cause : causes) {
this.beanFactory.onSuppressedException(cause);
}
throw ex;
}
throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Could not resolve matching constructor " +
"(hint: specify index/type/name arguments for simple parameters to avoid type ambiguities)");
}
/**
* 如果存在歧义的结构器汇合不为空,并且以后BeanDefinition为严格模式,那么则抛出异样,只有当BeanDefinition为宽松模式时,这种状况才不会抛异样
* */
else if (ambiguousConstructors != null && !mbd.isLenientConstructorResolution()) {
throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Ambiguous constructor matches found in bean '" + beanName + "' " +
"(hint: specify index/type/name arguments for simple parameters to avoid type ambiguities): " +
ambiguousConstructors);
}
/**
* 如果以后getBean没有传参数,那么则将以后的结构器和参数放到缓存外面去,可能Spring认为传入参数的状况下说不准咱们筹备怎么做,所以罗唆咱们本人传入参数的就不缓存了
* */
if (explicitArgs == null) {
argsHolderToUse.storeCache(mbd, constructorToUse);
}
}
/**
* 以下没什么好说的,new示例进去,存入beanWrapper中,return回去
* */
try {
final InstantiationStrategy strategy = beanFactory.getInstantiationStrategy();
Object beanInstance;
if (System.getSecurityManager() != null) {
final Constructor<?> ctorToUse = constructorToUse;
final Object[] argumentsToUse = argsToUse;
beanInstance = AccessController.doPrivileged((PrivilegedAction<Object>) () ->
strategy.instantiate(mbd, beanName, beanFactory, ctorToUse, argumentsToUse),
beanFactory.getAccessControlContext());
}
else {
beanInstance = strategy.instantiate(mbd, beanName, this.beanFactory, constructorToUse, argsToUse);
}
bw.setBeanInstance(beanInstance);
return bw;
}
catch (Throwable ex) {
throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Bean instantiation via constructor failed", ex);
}
}
正文出处:https://blog.51cto.com/u_1424…
办法十分的长,咱们总结一下能够归为:
-
结构函数参数的确定。
依据传入的explicitArgs参数判断
。如果 explicitArgs 不为空,则能够间接确定参数,因为 explicitArgs 是在调用 bean 的时候用户指定的,在 BeanFactory 类中存在这样的办法:Object getBean(String name,Object... args)
。所以如果不为空,则间接能够确定结构函数参数就是它。除此之外,如果结构函数参数曾经记录在缓存中,那么便能够间接拿来应用。须要留神的是:在缓存中拿到的可能是参数的最终类型也可能是初始类型,比方:参数要求是 int,而原始参数可能是 String,那么即便在缓存中失去参数,也须要进行类型转换器的过滤以确保参数类型与对应的结构函数参数齐全对应。如果以上两种状况都无奈确定。那么只能进一步剖析。通过 mbd.getConstructorArgumentValues()获取配置的构造函数信息,有了配置中的信息便能够获取对应的参数值信息,获取参数值的信息包含指定值,也可能是对另一个 bean 的援用,这个解决委托给了 resolveConstructorArguments 办法,并返回能解析到的参数的个数。
-
构造函数的确定。咱们确定了构造函数的参数后,就能够去依据参数去锁定对应的构造函数。匹配的办法就是依据参数个数去进行匹配,在匹配前须要先对构造函数依照 public 优先参数数量降序,非 public 结构函数参数数量降序。
因为还能够通过指定参数名称设定参数值的状况,如
<constructor-arg name="aa">
,那么这种状况首先要确定构造函数中的参数名称,有两种形式,一种通过注解的形式获取,另一种通过 Spring 中的工具类 ParameterNameDiscoverer 来获取。 - 依据确定的构造函数转换对应的参数类型。次要是应用 Spring 中提供的类型转换器或者用户提供的自定义类型转换器进行转换。
- 构造函数不确定性的验证。因为有时候即便构造函数、参数名称、参数类型、参数值都确定也不肯定就间接锁定构造函数,不同构造函数的参数为父子关系,所以这里最初又做一次验证。
- 依据实例化策略以及失去的构造函数及结构函数参数实例化 bean
解析instanitateBean
办法
咱们理解了带参数的办法后,咱们接着看一下无参的默认结构器实例化过程。
protected BeanWrapper instantiateBean(final String beanName, final RootBeanDefinition mbd) {
try {
Object beanInstance;
final BeanFactory parent = this;
if (System.getSecurityManager() != null) {
beanInstance = AccessController.doPrivileged((PrivilegedAction<Object>) () ->
getInstantiationStrategy().instantiate(mbd, beanName, parent),
getAccessControlContext());
} else {
beanInstance = getInstantiationStrategy().instantiate(mbd, beanName, parent);
}
BeanWrapper bw = new BeanWrapperImpl(beanInstance);
initBeanWrapper(bw);
return bw;
} catch (Throwable ex) {
throw new BeanCreationException(
mbd.getResourceDescription(), beanName, "Instantiation of bean failed", ex);
}
}
其实外面就是获取实例化的策略,而后进行 instantiate 实例化。应用父类SimpleInstantiationStrategy
的instantiate
。
- 实例化策略
咱们提到了屡次实例化策略,那这到底是什么呢?其实通过后面的剖析,咱们曾经失去了实例化的所有信息,其实能够间接应用反射来进行实例化对象,然而 Spring 却不是这样做的。
public Object instantiate(RootBeanDefinition bd, @Nullable String beanName, BeanFactory owner) {
// Don't override the class with CGLIB if no overrides.
//如果有须要笼罩或者动静替换的办法须要应用cglib进行动静代理,如果没有的话则间接应用反射的形式进行实例化
if (!bd.hasMethodOverrides()) {
Constructor<?> constructorToUse;
synchronized (bd.constructorArgumentLock) {
constructorToUse = (Constructor<?>) bd.resolvedConstructorOrFactoryMethod;
if (constructorToUse == null) {
final Class<?> clazz = bd.getBeanClass();
if (clazz.isInterface()) {
throw new BeanInstantiationException(clazz, "Specified class is an interface");
}
try {
if (System.getSecurityManager() != null) {
constructorToUse = AccessController.doPrivileged(
(PrivilegedExceptionAction<Constructor<?>>) clazz::getDeclaredConstructor);
}
else {
constructorToUse = clazz.getDeclaredConstructor();
}
bd.resolvedConstructorOrFactoryMethod = constructorToUse;
}
catch (Throwable ex) {
throw new BeanInstantiationException(clazz, "No default constructor found", ex);
}
}
}
return BeanUtils.instantiateClass(constructorToUse);
}
else {
// Must generate CGLIB subclass.
return instantiateWithMethodInjection(bd, beanName, owner);
}
}
CglibSubclassingInstantiationStrategy
public Object instantiate(@Nullable Constructor<?> ctor, Object... args) {
Class<?> subclass = createEnhancedSubclass(this.beanDefinition);
Object instance;
if (ctor == null) {
instance = BeanUtils.instantiateClass(subclass);
}
else {
try {
Constructor<?> enhancedSubclassConstructor = subclass.getConstructor(ctor.getParameterTypes());
instance = enhancedSubclassConstructor.newInstance(args);
}
catch (Exception ex) {
throw new BeanInstantiationException(this.beanDefinition.getBeanClass(),
"Failed to invoke constructor for CGLIB enhanced subclass [" + subclass.getName() + "]", ex);
}
}
// SPR-10785: set callbacks directly on the instance instead of in the
// enhanced class (via the Enhancer) in order to avoid memory leaks.
Factory factory = (Factory) instance;
factory.setCallbacks(new Callback[] {NoOp.INSTANCE,
new LookupOverrideMethodInterceptor(this.beanDefinition, this.owner),
new ReplaceOverrideMethodInterceptor(this.beanDefinition, this.owner)});
return instance;
}
首先在判断了如果 beanDefinition.getMethodOverrides()为空则阐明用户应用 replace 或者 lookup 的配置办法,那就间接应用反射的形式。在应用了这两种个性的话就只能应用动静代理。
记录创立 bean 的 ObjectFactory
接下来咱们持续看 doCreateBean 办法,这里进行了循环依赖查看等
// Eagerly cache singletons to be able to resolve circular references
// even when triggered by lifecycle interfaces like BeanFactoryAware.
//是否须要提前曝光,单例&&容许循环依赖&&以后bean正在创立中,检测循环依赖
boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
if (logger.isTraceEnabled()) {
logger.trace("Eagerly caching bean '" + beanName +
"' to allow for resolving potential circular references");
}
// 为防止前期循环依赖,能够在 bean 初始化实现前将创立实例的 objectFactory退出缓存
// 对bean 再一次依赖援用,次要利用SmartInstantiationAwareBeanPostProcessor
// 其中咱们相熟的AOP就是在这里将advice 动静织入,若没有间接返回bean
addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}
earlySingletonExposure
:提前曝光的实例mbd.isSingleton()
:是否为单例this.allowCircularReferences
:是否容许循环依赖isSingletonCurrentlyInCreation(beanName)
:该 bean 是否在创立中
这里就拿 TestA 依赖 TestB,TestB 依赖 TestA 举例。
咱们上一篇文章曾经具体讲过。这里就不再赘述。
发表回复