Spring底层核心原理

Bean创建过程

  • 利用该类的构造方法来实例化得到一个对象(但是如何一个类中有多个构造方法,Spring则会进行选择,这个叫做推断构造方法)
  • 得到一个对象后,Spring会判断该对象中是否存在被@Autowired注解了的属性,把这些属性找出来并由Spring进行赋值(依赖注入)
  • 依赖注入后,Spring会判断该对象是否实现了BeanNameAware接口、BeanClassLoadAware接口、BeanFactoryAware接口,如果实现了,就表示当前对象必须实现该接口中所定义的setBeanName()、setBeanCalssLoader()、setBeanFactory()方法,那Spring就会调用这些方法并传入相应的参数(Aware回调)
  • Aware回调后,Spring会判断该对象中是否存在某个方法被@PostConstruct注解了
  • 紧接着,Spring会判断该对象是否实现了InitializingBean接口,如果实现了,就表示当前对象必须实现该接口的afterPropertiesSet()方法,那Spring就会调用当前对象中的afterPropertiesSet()方法(初始化)
  • 最后,Spring会判断当前对象需不需要进行AOP,如果不需要那么Bean就创建完了,如果需要进行AOP,则会进行动态代理并生成一个代理对象做为Bean(初始化后)

通过最后一步,我们可以发现,当Spring根据UserService类来创建一个Bean时:

  • 如果不进行AOP,那么Bean就是UserService类的构造方法所得到的对象。
  • 如果需要进行AOP,那么Bean就是UserService的代理类所实例化得到的对象,而不是Userservice本身所得到的对象。

Bean对象创建出来后:

  • 如果当前Bean是单例Bean,那么会把该对象存入一个Map<String,Object>,Map的key为beanName,value为Bean对象。这样下次getBean时就可以直接从Map中拿到对应的Bean对象了。
  • 如果当前的Bean是原型Bean,那么后续没有其它动作,不会存入一个Map,下次getBean时会再次执行上述创建过程,得到一个新的Bean对象。

推断构造方法

Spring在基于某个生成Bean的过程中,需要利用该类的构造方法来实例化得到一个对象,但是如果一个类存在多个构造方法,Spring会使用哪个呢?

Spring的判断逻辑如下:

  • 如果一个类只存在一个构造方法,不管该构造方法是无参构造方法,还是有参构造方法,Spring都会用这个构造方法
  • 如果一个类存在多个构造方法
  • 这些构造方法中,存在一个无参的构造方法,那么Spring就会用这个无参的构造方法
  • 这些构造方法中,不存一个无参的构造方法,那么Spring就会报错

Spring的设计思想是这样的:

  • 如果一个类只有一个构造方法,那么没得选择,只能用这个构造方法
  • 如果一个类存在多个构造方法,Spring不知道如何选择,就会看是否有无参的构造方法,因为无参构造方法本身表示了一种默认的意义
  • 不过如果某个构造方法上加了@Autowired注解,那就表示程序员告诉Spring就用这个加了注解的方法,那Spring就会用这个加了@Autowired注解构造方法了

需要重视的是,如果Spring选择了一个有参的构造方法,Spring在调用这个有参构造方法时,需要传入参数,那这么个参数是怎么来的呢?

Spring会根据入参的类型和入参的名字去Spring中找Bean对象(以单例Bean为例,Spring会从单例池那个Map中去找)

  • 先根据入参类型找,如果只找到一个,那就直接用来作为入参
  • 如果根据类型找到多个,则再根据入参名字来确定唯一一个
  • 最终如果没有找到,则会报错,无法创建当前Bean对象