废话不多说,直接进入正题
看过上一篇文章的同学,应该都知道接下去是要讲什么了
没错,就是IOC最重要的一个过程,实例化bean
也就是finishBeanFactoryInitialization(beanFactory)这个方法
源码中的注释是 Instantiate all remaining (non-lazy-init) singletons.
翻译一下就是 : 实例化所有剩余的(非懒加载)单例,很好理解吧
好了,我们点进去,开始阅读
为了方便大家阅读,我把所有注释都翻译成了中文,如果英语比较好的同学, 建议看英文,毕竟翻过之后会少点意思。
这里我们主要看最后一个方法,实例化非懒加载的单例。
由于源码过长过多,我不会把所有代码都做截图,且不能面面俱到的讲述每一个细节。我会把主要的整个流程捋一遍,如果有疑问的同学欢迎评论私信探讨。
好了,进入这个方法之后
主要看getBean方法,其他我也用注释写了大家一看就知道了。这个方法一直跟进去,就到了doGetBean()方法
大家应该都知道,先是会去缓存里面找bean,如果没有,才会创建bean
创建bean的过程,spring根据scope做了区分,单例,多例,自定义scope,而大多数情况都是单例,所以我们看这个代码
该方法进去,继续跟进doCreateBean()方法,然后主要先看这个方法
这个方法主要是做通过反射拿到class对象,然后通过反射调用构造函数创建实例。值得一提的是,这里区分了正在实例化的bean中有@Autowird注解的构造函数的情况。那么大家可以想一想,如果构造函数上有@Autowird,你会怎么做?其实不难想明白,我相信大多数人想的,和spring源码肯定大致方向上是一致的。只不过spring有专门的构造函数解析器等封装的比较好的类,这里最麻烦的是构造函数参数的处理,这里不建议深究,先弄懂主流程。不过,需要知道的是,不论是构造函数还是属性,一旦有个@Autowird或者@Resource注解,都会触发beanFactory的getBean()方法,而这个方法,就是本文讲的内容,这里有点递归的意思。这也是spring解决循环依赖的基础。
初步创建出bean之后,是不是就该做属性注入了?在这之前,spring还做了一件事。
看过上一篇文章的同学应该还有印象,我说过你会在IOC源码中经常看到类似这样的一段代码,点进去
经典的postProcessor接口调用,我在注解上有些,这里有包括我们经常用的@PostConstruct,@PreDestroy,@Resource,@Autowired,@Value等注解的支持。具体大家可以点进对应的实现类中看到的实现过程细节。由于篇幅关系,实在不能每个点都涉及。
做完这些基础工作之后
接下去有3个很重要的方法,添加三级缓存,依赖注入,及实例化之后的调用。
关于添加三级缓存这个,我打算单独下一篇文章来讲一讲,顺便捋一捋循环依赖的怎么解决的。
这里我们先看第二个方法依赖注入
重点是这一块代码
我们选我们最熟悉的autowired来看看是怎么完成依赖注入的
spring源码常用手段,先收集再处理,这里先收集注解信息包装成注入元数据,这里就大家自行点进去看,思路其实和我们自己做也没什么区别,反射获取class信息,获取属性上的注解,和注解中的属性值,像autowired的required属性的value,然后去容器中找相对应的bean,反射赋值。
最后再看第三个方法
这里完成了一些bean实例化之后的事情,比如Aware(感知接口,我一般这么叫)接口的方法调用,和之前提过的后置处理器的调用,像我们常用的@PostContruct注解,在初始化完成后要调用的方法。值得一提的是,这里也是AOP代理类生成的入口。
我们不妨猜一下,它是怎么做的。
首先,bean已经创建完成了,然后怎么判断它是否需要生成代理类呢?就是看是否有切面有切点。也就是aop中基础的2个关键词,advice和pointCut。
如果需要生成代理类,则生成,具体怎么生成,之后看到AOP的源码时会讲。面试的时候,你也不能光说,动态代理,JDK动态代理和CGLIB2种,然后就没了。
那么IOC创建实例的整个大过程,差不多就是这样了,其中有很多的细节,具体做了什么,确实因为文字的原因,无法面面俱到,我相信看到这篇文章的同学, 肯定是处于学习阶段,我希望大家一定要多跟几遍源码,自己多想多看。
如果有什么不对的地方,或者有什么疑问,欢迎评论指正探讨。
下一篇文章,应该会着重讲一讲spring的三级缓存和如何解决循环依赖的。