懒实例化
bean
ApplicationContext
的默认行为是在启动的时候,尽可能早的预实例化所有的单例
bean
。所谓“预实例化”是指
ApplicationContext
实例会在自身的初始化过程中,尽可能的创建和配置所有的单例
bean,预实例化阶段只是简单列举所用单态(默认状态)Bean,给每个包含依赖注入的Bean创建实例,并缓存该实例
。通常这是正确的,因为这意味着在配置中,或者在相关辅助环境中的错误会被立刻发现(相比可能会在数小时甚至数天才被发现而言)。
然后,有时这种默认的行为却不是我们所需要的。当你不需要
ApplicationContext
预实例化单例
bean
时,你可以(在
bean
定义文件的基础上)有选择地控制这种默认行为,将
bean
设置为懒实例化(
lazy-initialized
)。懒实例化
bean
告诉容器实例在启动时创建,还是当被请求的时候才创建。例如:我们这个项目就是采用预实例化Bean,从控制台可以看到Tomcat在加载Bean定义档时,就实例化了Bean(通过构造函数),如CustomBaseFormController,LoginController等。
通过
XML
配置
bean
时,是通过
'lazy-init'
属性来控制“懒加载”(
lazy-loading
)的。参看下面的例子:
<bean id="lazy" class="com.foo.ExpensiveToCreateBean" lazy-init="true">
<!-- various properties here... -->
</bean>
<bean name="not.lazy" class="com.foo.AnotherBean">
<!-- various properties here... -->
</bean>
对于上面的配置,叫做
’lazy’
的
bean
在
ApplicationContext
初始化时不会被预实例化,而叫做
’not.lazy’
的
bean
则会被尽早的预实例化。
关于懒实例化,需要说明的一点是,即使
bean
被配置成懒实例化,假如该
bean
是一个非懒实例化的
bean
的依赖,那么在
ApplicationContext
预实例化
bean
的时候,会创建它的所有依赖,即使它们中存在懒实例化的
bean
!所以,当容器把你配置成懒实例化的
bean
实例化的时候,不必疑惑;那是因为,这个懒实例化的
bean
是被依赖注射到了你的配置中其他某个地方的非懒实例化的单例
bean
。
在容器层次,还可以通过使用
<beans/>
元素的
'default-lazy-init'
属性实现懒实例化;参看下面的例子:
<beans default-lazy-init="true">
<!-- no beans will be eagerly pre-instantiated... -->
</beans>