ApplicationContext 容器的默认行为是在启动服务器时将所有 singleton bean提前进?实例化。提前实例化意味着作为初始化过程的?部分,ApplicationContext 实例会创建并配置所有的singleton bean。 控制bean是否要延迟加载由lazy-init属性来控制,默认为false。
<!-- lazy-init属性默认为false-->
<bean id="lazyBean" class="com.nanmao.LazyBean" lazy-init="false"/>
我们可以通过在启动容器之后进行debug打点来进行查看。如下图,applicationContext -> beanFactory -> singletonObjects,我们可以看到在singletonObjects中共有12个已创建的单例bean对象,包括lazyBean。是一个ConcurrentHashMap类型。
lazy-init=“false”,?即加载,表示在spring启动时,?刻进?实例化。 如果不想让?个singleton bean 在 ApplicationContext实现初始化时被提前实例化,那么可以将bean设置为延迟实例化。设置 lazy-init 为 true 的 bean 将不会在 ApplicationContext 启动时提前被实例化,?是第?次向容器通过 getBean 索取 bean 时实例化的。如果?个设置了立即加载的 bean1,引用了?个延迟加载的 bean2 ,那么 bean1 在容器启动时被实例化,而bean2 由于被 bean1 引用,所以也被实例化,这种情况也符合延时加载的 bean 在第?次调用时才被实例化的规则。 如下图,当lazy-init属性为true时,在容器初始化后,singletonObjects中不存在lazyBean。
<bean id="lazyBean" class="com.nanmao.LazyBean" lazy-init="true"/>
也可以在容器层次中通过在元素上使? “default-lazy-init” 属性来控制全局的延时初始化。如果和bean标签中lazy-init属性同时配置,lazy-init优先级更高。如下?配置:
<?xml version="1.0" encoding="UTF-8"?>
<beans default-lazy-init="true" xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
如果?个 bean 的 scope 属性为 scope=“pototype” 时,即使设置了 lazy-init=“false”,容器启动时也不 会实例化bean,?是调? getBean 方法实例化的。spring容器不管理多例bean对象。 注解: @lazy 注解到类上,默认值为true,表明此bean对象要延迟加载
@Target({ElementType.TYPE, ElementType.METHOD, ElementType.CONSTRUCTOR, ElementType.PARAMETER, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Lazy {
boolean value() default true;
}
应用场景: (1)开启延迟加载?定程度提?容器启动和运转性能 (2)对于不常使?的 Bean 设置延迟加载,这样偶尔使?的时候再加载,不必要从?开始该 Bean 就占?资源。
|