Bean的实例化过程
- 一、实例化FactoryMethod方法对应的实例
- 二、实例化带有@Autowired注解的有参构造方法
- 三、实例化没有@Autowired注解的有参构造方法
- 四、实例化无参构造方法
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,直接执行无参构造方法的实例化。