Bean的实例化从finishBeanFactoryInitialization进入,循环beanDefinitionNames容器中的beanName。把父类属性复制到子类中,根据BeanDefinition判断bean是否是抽象的、单例的、懒加载的。

从getBean来到doGetBean,先从缓存中拿对象,拿到了就做类型转换然后返回。如果缓存中没有,则来到Bean的实例化过程,dependsOn属性中的类先实例化(如果这个类有循环依赖则报错)。
着重看下面部分的代码。

来到getSingleton方法,首先从一级缓存中拿(一级缓存中是已经实例化的Bean),拿不到就走下面的代码。beforeSingletonCreation是把正在创建的对象名称放到set容器SingletonsCurrentlyInCreation。

getObject是具体的实例化过程,这是一个函数式接口的方法。

这里的 Lambda 表达式就是getObject的实现。所以接下来看createBean方法。
sharedInstance = getSingleton(beanName, () -> {
try {
return createBean(beanName, mbd, args);
}
从createBean来到doCreateBean,接着从doCreateBean来到createBeanInstance。createBeanInstance的返回值类型BeanWrapper是一个包装类,包装了创建好的bean。在createBeanInstance中,可以分析出Bean的实例化有四种情况。
一、实例化FactoryMethod方法对应的实例
分两种情况。
一种是定义了factory-bean,factory-bean的factory-method是非静态方法,在方法中实例化bean。


另一种没有定义factory-bean,定义了class属性,factory-method为这个类中定义的实例化方法,是一个静态方法。


来看Spring源码中的createBeanInstance方法是怎么对这两种情况解析的。

判断factoryBeanName属性是否为空。如果不为空,实例化factoryBean,将isStatic设置为false。如果为空,拿到bean的class,设置isStatic为true。

判断factoryBean中的实例化方法是否与xml文件中定义的factory-method方法名称相同,且为非静态。或者beanClass中的方法是否与xml文件中定义的factory-method方法名称相同,且为静态。如果满足条件,则加入到容器中。

最后,在instantiate中可以看到,会调用factoryMethod.invoke()实例化Bean。

二、实例化带有@Autowired注解的有参构造方法

首先看createBeanInstance的determineConstructorsFromBeanPostProcessors,循环找到在registerBeanPostProcessors实例化过的AutowiredAnnotationBeanPostProcessor,由这个类来解析带有@Autowired注解的有参构造方法。然后执行determineCandidateConstructors方法。

判断是否有@Autowired注解,获取@Autowired注解required的值。如果required为true,且candidates这个容器是空的,则加入到candidates。
当有多个带有@Autowired注解,required都为true的构造方法时,if (requiredConstructor != null)这里会抛出异常。因为只要有一个构造方法的required为true,requiredConstructor就不是null。
当有两个带有@Autowired注解,required分别为true和false的构造方法时,if (!candidates.isEmpty())这里会抛出异常,因为不管required是true还是false,都会加入candidates容器,容器不为空就会抛出异常。
当有多个带有@Autowired注解,required都为false的构造方法时,这些构造方法都会被加入candidates容器并返回。

将构造方法返回,进入autowireConstructor,如果有多个构造方法,会按照参数个数排序,参数个数多的在前。仅把第一个构造方法拿出来执行bw.setBeanInstance(instantiate(beanName, mbd, constructorToUse, argsToUse));

在instantiate的BeanUtils.instantiateClass(ctor, args);中可以看到,这里是执行了ctor.newInstance()去实例化Bean。

三、实例化没有@Autowired注解的有参构造方法
在AutowiredAnnotationBeanPostProcessor.determineCandidateConstructors中,此时candidates容器是空的,如果只有一个有参构造方法,满足下图else if的条件,于是返回了包含这个有参构造方法的对象,最后执行ctor.newInstance。

当没有定义无参构造方法,且有多个有参构造方法时,满足下图的条件,没有构造方法返回。

此时,返回的对象值是null。

于是createBeanInstance中会来到无参构造方法的实例化。

在instantiate方法中会拿到类的无参构造方法去实例化。可是根本就没有定义无参构造方法,会报错。

四、实例化无参构造方法
当没有定义有参构造方法,只有默认的无参构造方法时。AutowiredAnnotationBeanPostProcessor.determineCandidateConstructors会返回null,直接执行无参构造方法的实例化。
Spring Bean实例化详解:工厂方法、构造器与Autowired
4803

被折叠的 条评论
为什么被折叠?



