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

        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,直接执行无参构造方法的实例化。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值