factoryMethod方式实例化
在xml时代,我们有两种使用factoryMethod实例化bean的方式: 样例1:
<bean id="testObj1Factory" class="com.example.spring.beans2.TestObj1Factory"/>
<bean id="testObj1" factory-bean="testObj1Factory" factory-method="createTestObj"/>
样例2:
<bean id="testObj2" class="com.example.spring.beans2.TestObj2" factory-method="createInstance"/>
这两种方式都是使用factory-method 的方式对bean进行实例化。但是他们是有区别的: 样例1中需要先创建一个factoryBean,然后需要被实例化的bean上配置factory-bean 和factory-method ,相当于调用factoryBean的factoryMethod方法后创建bean对象。 样例2中并没有配置factoryBean而是配置了一个class 属性和一个factory-method ,这时,这个factory-method就必须是静态方法,因为只有静态方法可以在没有实例化对象时进行调用。 但是现在已经很少有人使用xml配置bean了。但是通过factory-method 来实例化bean的方式还是保留了下来,比如我们常用的@Bean注解也是基于factoryMethod的方式来实例化的。
有参和无参构造函数实例化
- 如果一个bean中,只有一个构造器,那么不管这个构造器上是否有@Autowired注解,Spring都会通过这个构造器来实例化当前bean。
- 如果当前bean中存在多个被@Autowired修饰的构造器时,那么必须都是@Autowired(required=false),否则会抛异常,如果都是@Autowired(required=false)Spring会优先使用参数个数最多那个构造器来初始化bean。
    - 如果当前bean中的构造器有多个而且都没有被@Autowired修饰,那么Spring会走bean的无参构造函数,如果bean没有提供无参构造器,那么会抛出异常

bean的实例化流程
所有bean的实例化都是从org.springframework.beans.factory.config.ConfigurableListableBeanFactory#preInstantiateSingletons 开始的  ps:其实Spring的注释写的还挺好的。 在实例化时,特意标注了,只会实例化非抽象、单例、非LazyInit的bean  逻辑一直往下走,会走到org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean 方法,最后会执行到这里,这个createBean 就是创建单例bean了  然后会执行到org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#doCreateBean  所有的bean的定义信息都会被封装到BeanDefinition中,与之类似的,bean也会被封装到一个BeanWrapper 的对象中。可以认为创建BeanWrapper的过程就是一个bean实例化的过程。BeanWrapper是由org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#createBeanInstance 来创建的,如果这个BeanDefinition中的getFactoryMethodName() 不为空,那么就会通过instantiateUsingFactoryMethod 来初始化一个bean  
其实bean的实例化感觉也没有很神秘,最后也是通过反射,要么调用某个方法,要么调用构造器
@Autowired@Value @Resource等注解收集
这些注解的收集是在bean的实例化之后进行的  当createBeanInstance之后,会执行一个叫applyMergedBeanDefinitionPostProcessors 的方法,这个方法是一个BeanPostProcessor 接口的应用  其实收集的原理的也很简单,就是反射,看有没有我们关心的注解,这唯一有点意思的是,Spring提供了一个反射用的util,可以通过这个util直接遍历字段和方法  需要额外注意的是,当你处理通过反射获得的方法时,需要获取到这个方法的桥接方法,然后用桥接方法再去找注解,不然可能会找不到注解。
|