Bean的实例化过程

  • 一、实例化FactoryMethod方法对应的实例
  • 二、实例化带有@Autowired注解的有参构造方法
  • 三、实例化没有@Autowired注解的有参构造方法
  • 四、实例化无参构造方法


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

spring重新实例化bean spring实例化bean过程_构造方法

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

        着重看下面部分的代码。

spring重新实例化bean spring实例化bean过程_源码_02

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

spring重新实例化bean spring实例化bean过程_源码_03


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

spring重新实例化bean spring实例化bean过程_构造方法_04


        这里的 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。

spring重新实例化bean spring实例化bean过程_构造方法_05


spring重新实例化bean spring实例化bean过程_java_06

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

spring重新实例化bean spring实例化bean过程_spring_07


spring重新实例化bean spring实例化bean过程_构造方法_08


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

spring重新实例化bean spring实例化bean过程_源码_09


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

spring重新实例化bean spring实例化bean过程_spring重新实例化bean_10


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

spring重新实例化bean spring实例化bean过程_spring重新实例化bean_11


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

spring重新实例化bean spring实例化bean过程_构造方法_12

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

spring重新实例化bean spring实例化bean过程_源码_13

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

spring重新实例化bean spring实例化bean过程_spring重新实例化bean_14


        判断是否有@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容器并返回。

spring重新实例化bean spring实例化bean过程_java_15


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

spring重新实例化bean spring实例化bean过程_spring_16


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

spring重新实例化bean spring实例化bean过程_源码_17

三、实例化没有@Autowired注解的有参构造方法

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

spring重新实例化bean spring实例化bean过程_java_18


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

spring重新实例化bean spring实例化bean过程_构造方法_19


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

spring重新实例化bean spring实例化bean过程_spring重新实例化bean_20

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

spring重新实例化bean spring实例化bean过程_java_21


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

spring重新实例化bean spring实例化bean过程_spring_22

四、实例化无参构造方法

        当没有定义有参构造方法,只有默认的无参构造方法时。AutowiredAnnotationBeanPostProcessor.determineCandidateConstructors会返回null,直接执行无参构造方法的实例化。