Spring系列第12篇:lazy-init:bean延迟初始化

star2017 1年前 ⋅ 657 阅读

bean初始化的2种方式

  1. 实时初始化
  2. 延迟初始化

bean实时初始化

在容器启动过程中被创建组装好的bean,称为实时初始化的bean,spring中默认定义的bean都是实时初始化的bean,这些bean默认都是单例的,在容器启动过程中会被创建好,然后放在spring容器中以供使用。

实时初始化bean的有一些优点

  1. 更早发下bean定义的错误:实时初始化的bean如果定义有问题,会在容器启动过程中会抛出异常,让开发者快速发现问题
  2. 查找bean更快:容器启动完毕之后,实时初始化的bean已经完全创建好了,此时被缓存在spring容器中,当我们需要使用的时候,容器直接返回就可以了,速度是非常快的

案例

ActualTimeBean类

  1. package com.javacode2018.lesson001.demo11;
  2. /**
  3. * 实时初始化的bean
  4. */
  5. public class ActualTimeBean {
  6. public ActualTimeBean() {
  7. System.out.println("我是实时初始化的bean!");
  8. }
  9. }

一会我们在spring中创建上面这个对象,构造函数中会输出一段话,这段话会在spring容器创建过程中输出。

actualTimeBean.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">
  6. <bean id="actualTimeBean" class="com.javacode2018.lesson001.demo11.ActualTimeBean"/>
  7. </beans>

测试用例

  1. package com.javacode2018.lesson001.demo11;
  2. import org.junit.Test;
  3. import org.springframework.context.support.ClassPathXmlApplicationContext;
  4. /**
  5. * 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!
  6. * bean默认是实时初始化的,可以通过bean元素的lazy-init="true"将bean定义为延迟初始化
  7. */
  8. public class LazyBeanTest {
  9. @Test
  10. public void actualTimeBean() {
  11. System.out.println("spring容器启动中...");
  12. String beanXml = "classpath:/com/javacode2018/lesson001/demo11/actualTimeBean.xml";
  13. new ClassPathXmlApplicationContext(beanXml); //启动spring容器
  14. System.out.println("spring容器启动完毕...");
  15. }
  16. }

注意上面代码,容器启动前后有输出,运行actualTimeBean输出:

  1. spring容器启动中...
  2. 我是实时初始化的bean!
  3. spring容器启动完毕...

可以看出actualTimeBean这个bean是在容器启动过程中被创建好的。

延迟初始化的bean

从上面我们可以看出,实时初始化的bean都会在容器启动过程中创建好,如果程序中定义的bean非常多,并且有些bean创建的过程中比较耗时的时候,会导致系统消耗的资源比较多,并且会让整个启动时间比较长,这个我估计大家都是有感受的,使用spring开发的系统比较大的时候,整个系统启动耗时是比较长的,基本上多数时间都是在创建和组装bean。

spring对这些问题也提供了解决方案:bean延迟初始化。

所谓延迟初始化,就是和实时初始化刚好相反,延迟初始化的bean在容器启动过程中不会创建,而是需要使用的时候才会去创建,先说一下bean什么时候会被使用:

  1. 被其他bean作为依赖进行注入的时候,比如通过property元素的ref属性进行引用,通过构造器注入、通过set注入、通过自动注入,这些都会导致被依赖bean的创建
  2. 开发者自己写代码向容器中查找bean的时候,如调用容器的getBean方法获取bean。

上面这2中情况会导致延迟初始化的bean被创建。

延迟bean的配置

在bean定义的时候通过lazy-init属性来配置bean是否是延迟加载,true:延迟初始化,false:实时初始化

  1. <bean lazy-init="是否是延迟初始化" />

我们来2个案例看一下效果。

案例1

LazyInitBean类

  1. package com.javacode2018.lesson001.demo11;
  2. public class LazyInitBean {
  3. public LazyInitBean() {
  4. System.out.println("我是延迟初始化的bean!");
  5. }
  6. }

lazyInitBean.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">
  6. <bean id="lazyInitBean" class="com.javacode2018.lesson001.demo11.LazyInitBean" lazy-init="true"/>
  7. </beans>

注意上面的lazy-init="true"表示定义的这个bean是延迟初始化的bean。

测试用例

LazyBeanTest中加个方法

  1. @Test
  2. public void lazyInitBean() {
  3. System.out.println("spring容器启动中...");
  4. String beanXml = "classpath:/com/javacode2018/lesson001/demo11/lazyInitBean.xml";
  5. ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(beanXml); //启动spring容器
  6. System.out.println("spring容器启动完毕...");
  7. System.out.println("从容器中开始查找LazyInitBean");
  8. LazyInitBean lazyInitBean = context.getBean(LazyInitBean.class);
  9. System.out.println("LazyInitBean:" + lazyInitBean);
  10. }

注意上面的输出,容器启动前后有输出,然后又从容器中查找LazyInitBean。

运行输出

执行lazyInitBean方法,输出:

  1. spring容器启动中...
  2. spring容器启动完毕...
  3. 从容器中开始查找LazyInitBean
  4. 我是延迟初始化的bean!
  5. LazyInitBean:com.javacode2018.lesson001.demo11.LazyInitBean@11dc3715

代码结合输出可以看出来,LazyInitBean在容器启动过程中并没有创建,当我们调用context.getBean方法的时候,LazyInitBean才被创建的。

案例2

上面这种方式是我们主动从容器中获取bean的时候,延迟初始化的bean才被容器创建的,下面我们再来看一下当延迟初始化的bean被其他实时初始化的bean依赖的时候,是什么时候创建的。

ActualTimeDependencyLazyBean类

  1. package com.javacode2018.lesson001.demo11;
  2. public class ActualTimeDependencyLazyBean {
  3. public ActualTimeDependencyLazyBean() {
  4. System.out.println("ActualTimeDependencyLazyBean实例化!");
  5. }
  6. private LazyInitBean lazyInitBean;
  7. public LazyInitBean getLazyInitBean() {
  8. return lazyInitBean;
  9. }
  10. public void setLazyInitBean(LazyInitBean lazyInitBean) {
  11. this.lazyInitBean = lazyInitBean;
  12. System.out.println("ActualTimeDependencyLazyBean.setLazyInitBean方法!");
  13. }
  14. }

ActualTimeDependencyLazyBean类中有个lazyInitBean属性,对应的有get和set方法,我们将通过set方法将lazyInitBean对象注入。

actualTimeDependencyLazyBean.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">
  6. <bean id="lazyInitBean" class="com.javacode2018.lesson001.demo11.LazyInitBean" lazy-init="true"/>
  7. <bean id="actualTimeDependencyLazyBean" class="com.javacode2018.lesson001.demo11.ActualTimeDependencyLazyBean">
  8. <property name="lazyInitBean" ref="lazyInitBean"/>
  9. </bean>
  10. </beans>

注意上面定义了2个bean:

lazyInitBean:lazy-init为true,说明这个bean是延迟创建的

actualTimeDependencyLazyBean:通过property元素来注入lazyInitBean,actualTimeDependencyLazyBean中没有指定lazy-init,默认为false,表示是实时创建的bean,会在容器创建过程中被初始化

测试用例

LazyBeanTest中加个方法,如下:

  1. @Test
  2. public void actualTimeDependencyLazyBean() {
  3. System.out.println("spring容器启动中...");
  4. String beanXml = "classpath:/com/javacode2018/lesson001/demo11/actualTimeDependencyLazyBean.xml";
  5. ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(beanXml); //启动spring容器
  6. System.out.println("spring容器启动完毕...");
  7. }

运行输出

  1. spring容器启动中...
  2. ActualTimeDependencyLazyBean实例化!
  3. 我是延迟初始化的bean!
  4. ActualTimeDependencyLazyBean.setLazyInitBean方法!
  5. spring容器启动完毕...

从容器中可以看到,xml中定义的2个bean都在容器启动过程中被创建好了。

有些朋友比较迷茫,lazyInitBean的lazy-init为true,怎么也在容器启动过程中被创建呢?

由于actualTimeDependencyLazyBean为实时初始化的bean,而这个bean在创建过程中需要用到lazyInitBean,此时容器会去查找lazyInitBean这个bean,然后会进行初始化,所以这2个bean都在容器启动过程中被创建的。

总结

延迟初始化的bean无法在程序启动过程中迅速发现bean定义的问题,第一次获取的时候可能耗时会比较长。在实际工作中用的比较少,作为了解,以后碰到的时候会有个印象。

案例源码

  1. 链接:https://pan.baidu.com/s/1p6rcfKOeWQIVkuhVybzZmQ
  2. 提取码:zr99

最新资料

更多内容请访问:IT源点

相关文章推荐

全部评论: 0

    我有话说: