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

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门,即可获取!
}

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

spring容器启动中…

我是实时初始化的bean!

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:实时初始化

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

案例1

LazyInitBean类

package com.javacode2018.lesson001.demo11;

public class LazyInitBean {

public LazyInitBean() {

System.out.println(“我是延迟初始化的bean!”);

}

}

lazyInitBean.xml
<?xml version="1.0" encoding="UTF-8"?>

<beans 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-4.3.xsd">

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

测试用例

LazyBeanTest中加个方法

@Test

public void lazyInitBean() {

System.out.println(“spring容器启动中…”);

String beanXml = “classpath:/com/javacode2018/lesson001/demo11/lazyInitBean.xml”;

ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(beanXml); //启动spring容器

System.out.println(“spring容器启动完毕…”);

System.out.println(“从容器中开始查找LazyInitBean”);

LazyInitBean lazyInitBean = context.getBean(LazyInitBean.class);

System.out.println(“LazyInitBean:” + lazyInitBean);

}

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

运行输出

执行lazyInitBean方法,输出:

spring容器启动中…

spring容器启动完毕…

从容器中开始查找LazyInitBean

我是延迟初始化的bean!

LazyInitBean:com.javacode2018.lesson001.demo11.LazyInitBean@11dc3715

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

案例2

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

ActualTimeDependencyLazyBean类

package com.javacode2018.lesson001.demo11;

public class ActualTimeDependencyLazyBean {

public ActualTimeDependencyLazyBean() {

System.out.println(“ActualTimeDependencyLazyBean实例化!”);

}

private LazyInitBean lazyInitBean;

public LazyInitBean getLazyInitBean() {

return lazyInitBean;

}

public void setLazyInitBean(LazyInitBean lazyInitBean) {

this.lazyInitBean = lazyInitBean;

System.out.println(“ActualTimeDependencyLazyBean.setLazyInitBean方法!”);

}

}

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

actualTimeDependencyLazyBean.xml
<?xml version="1.0" encoding="UTF-8"?>

<beans 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-4.3.xsd">

注意上面定义了2个bean:

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

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

测试用例

LazyBeanTest中加个方法,如下:

@Test

public void actualTimeDependencyLazyBean() {

System.out.println(“spring容器启动中…”);

String beanXml = “classpath:/com/javacode2018/lesson001/demo11/actualTimeDependencyLazyBean.xml”;

ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(beanXml); //启动spring容器

System.out.println(“spring容器启动完毕…”);

}

运行输出

spring容器启动中…

ActualTimeDependencyLazyBean实例化!

我是延迟初始化的bean!

ActualTimeDependencyLazyBean.setLazyInitBean方法!

spring容器启动完毕…

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

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

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

总结

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

案例源码


链接:https://pan.baidu.com/s/1p6rcfKOeWQIVkuhVybzZmQ

提取码:zr99

Spring系列


  1. Spring系列第1篇:为何要学spring?

  2. Spring系列第2篇:控制反转(IoC)与依赖注入(DI)

  3. Spring系列第3篇:Spring容器基本使用及原理

  4. Spring系列第4篇:xml中bean定义详解(-)

  5. Spring系列第5篇:创建bean实例这些方式你们都知道?

  6. Spring系列第6篇:玩转bean scope,避免跳坑里!

  7. Spring系列第7篇:依赖注入之手动注入

  8. Spring系列第8篇:自动注入(autowire)详解,高手在于坚持

最后

这份《“java高分面试指南”-25分类227页1000+题50w+字解析》同样可分享给有需要的朋友,感兴趣的伙伴们可挑战一下自我,在不看答案解析的情况,测试测试自己的解题水平,这样也能达到事半功倍的效果!(好东西要大家一起看才香)

image

image

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门,即可获取!
)

  1. Spring系列第6篇:玩转bean scope,避免跳坑里!

  2. Spring系列第7篇:依赖注入之手动注入

  3. Spring系列第8篇:自动注入(autowire)详解,高手在于坚持

最后

这份《“java高分面试指南”-25分类227页1000+题50w+字解析》同样可分享给有需要的朋友,感兴趣的伙伴们可挑战一下自我,在不看答案解析的情况,测试测试自己的解题水平,这样也能达到事半功倍的效果!(好东西要大家一起看才香)

[外链图片转存中…(img-VaOWnXvn-1714766048437)]

[外链图片转存中…(img-sA2eBDvI-1714766048438)]

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门,即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值