首先来看一段代码,看过上一节的朋友肯定对这段代码并不陌生。这一段代码诠释了Spring加载bean的完整过程,包括读取配置文件,扫描包,加载类,实例化bean,注入bean属性依赖。
1 | public void refresh() throws BeansException, IllegalStateException { |
上一节介绍了Spring是如何加载class文件的,本节主要围绕finishBeanFactoryInitialization(beanFactory)
方法,聊聊Spring是如何实例化bean的,从上面代码片段中的注解不难看出,此方法主要的任务就是实例化非懒加载的单例bean。闲话少叙,看代码。
1 | protected void finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory) { |
上面代码主要看最后一句beanFactory.preInstantiateSingletons()
。
1 | public void preInstantiateSingletons() throws BeansException { |
此方法首先将加载进来的beanDefinitionNames循环分析,如果是我们自己配置的bean就会走else
中的getBean(beanName)
,接着看。
1 | @Override |
doGetBean
方法内容太多,一段一段看。
1 | protected <T> T doGetBean( |
这里主要看Object sharedInstance = getSingleton(beanName)
。
1 | protected Object getSingleton(String beanName, boolean allowEarlyReference) { |
这里能看到,Spring会把实例化好的bean存入singletonObjects
,这是一个ConcurrentHashMap
,
1 | private final Map<String, Object> singletonObjects = new ConcurrentHashMap<String, Object>(64); |
当然这里我们bean并未实例化过,所以这里应该也不能get出什么东西来,也就是返回null了。if
子句也就不会执行了。那么接着看else
子句的内容。
1 | else { |
这两条验证也都不会实现,接写来就是重点了。
1 | try { |
在这里拿到RootBeanDefinition
并check,并获得bean的依赖,并循环迭代实例化bean。例如class A依赖于class B,就会先实例化B。下面的if ... else ...
就是真正实例化bean的地方。其实真正实例化bean的方法是createBean(beanName, mbd, args)
,只是区分了isSingleton
或isPrototype
,两者的区别在于,单例的(Singleton)被缓存起来,而Prototype是不用缓存的。首先看一下createBean(beanName, mbd, args)
。createBean
方法中除了做了一些实例化bean前的检查准备工作外,最核心的方法就是
1 | Object beanInstance = doCreateBean(beanName, mbd, args); |
由于这个过程涉及到的代码都是一大坨,就不贴出所有代码了。
1 | BeanWrapper instanceWrapper = null; |
首先就是创建一个bean的实例且封装到BeanWrapper
中,在这里bean已经实例化了。具体的实现方法是在org.springframework.beans.factory.support.SimpleInstantiationStrategy.instantiate(RootBeanDefinition beanDefinition, String beanName, BeanFactory owner)
中。
1 | @Override |
在这里不难看出实例化分两种情况,如果没有无参构造器是就生成CGLIB子类,否则就直接反射成实例。
1 | public static <T> T instantiateClass(Constructor<T> ctor, Object... args) throws BeanInstantiationException { |
既然已经有了实例对象了,那么,Spring是如何将bean的属性注入到bean的呢?返回到上面的doCreateBean
方法中。往下看找到populateBean(beanName, mbd, instanceWrapper);
,内幕就在这里。只贴部分代码:
1 | boolean hasInstAwareBpps = hasInstantiationAwareBeanPostProcessors(); |
这里是调用InstantiationAwareBeanPostProcessor
的具体子类的ibp.postProcessPropertyValues
方法注入属性。当我们使用@Resource
注解的时候,具体的子类是CommonAnnotationBeanPostProcessor
;如果使用的是@Autowired
注解,则具体的子类是AutowiredAnnotationBeanPostProcessor
。此方法内是委托InjectionMetadata
对象来完成属性注入。
1 | @Override |
findAutowiringMetadata
方法能拿到使用了特定注解的属性(Field)、方法(Method)及依赖的关系保存到checkedElements
集合<Set>
里,然后再执行自己的inject
方法。
1 | public void inject(Object target, String beanName, PropertyValues pvs) throws Throwable { |
真正干事的还是InjectedElement
的inject
方法。
1 | @Override |
其实别看代码这么多,最关键的部分就是:
1 | if (value != null) { |
在这里也就真相大白了,就是通过JDK反射特性,直接set值的。