这是我参与更文挑战的第7天,活动详情查看:更文挑战
在已经可以借助于BeanFactoryPostProcessor来干预容器的启动阶段的活动之后,我们就可以开始探索下一个阶段,即bean实例化阶段的实现逻辑了。
容器启动之后被不会马上就实例化相应的bean定义。容器现在仅仅拥有所有对象的BeanDefinition来保存实例化阶段将要用的必要信息。只有当请求方通过BeanFactory的getBean方法来请求某个对象实例的时候,才有可能触发Bean的实例化阶段的活动。BeanFactory的getBean方法可以被客户端对象显示调用,也可以在容器内部隐式调用。隐式调用有如下两种情况:
1.对于BeanFactory来说,对象实例化默认采用延迟初始化。通常情况下,当对象A被请求而需要第一次实例化的时候,如果它所依赖的对象B之前同样没有被实例化,那么容器会先实例化对象A所依赖的对象。这种情况是容器内部调用getBean方法,对于本次请求是隐式的。
2.ApplicationContext启动之后会实例化所有的bean定义。但ApplicationContext在实现的过程中依然遵循Spring容器实现流程的两个阶段,只不过它会在启动阶段的活动完成之后,紧接着调用注册到该容器的所有bean定义的实例化方法getBean()。具体可以查看refresh()
之前之所以说getBean()方法是有可能触发Bean实例化的,是因为只有当对应某个bean定义的getBean()方法第一次被调用时,不管是隐式还是显示的,Bean实例化才会被触发,第二次被调用则会直接返回容器缓存中的第一次实例化完的对象实例(prototype类型除外)。当getBean()方法内部发现该bean之前没有被实例化,则会通过createBean()方法来进行具体的对象实例化。实例化过程如下图所示:
未完待续…