Spring源码(三):Bean的实例化过程

142 阅读4分钟

本文已参与「新人创作礼」活动,一起开启掘金创作。

Spring源码(三):Bean的实例化过程

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

image.png

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

image.png

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

image.png

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

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

image.png

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

image.png image.png

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

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

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

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

image.png

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

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

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

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

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

image.png

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

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

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

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

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

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

image.png

四、实例化无参构造方法

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

———————————————— 版权声明:本文为CSDN博主「自然醒zzz」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 原文链接:blog.csdn.net/JustPlayCod…