首页 > 其他分享 >一不小心,你就掉进了Spring延迟初始化的坑!

一不小心,你就掉进了Spring延迟初始化的坑!

时间:2023-05-14 19:32:32浏览次数:35  
标签:初始化 Lazy Spring MyBean Bean myBean public 一不小心

前言

  书接上回,之前我们有聊到 Spring 的延迟初始化机制,是什么,有什么作用。今天跟各位大佬分享一下,我在使用 Spring 延迟初始化踩过的一些坑。

List<坑> 坑列表 = new ArrayList<>(2);

  首先,让我们回顾一下 Spring 延迟初始化的概念。在 Spring 中,延迟初始化指的是将 Bean 的实例化推迟到第一次被使用时,而不是在应用程序启动时就立即创建所有的 Bean。这种延迟加载的机制可以提高应用程序的性能和资源利用率。

坑 1. 延迟加载失效,被非延迟初始化的 Bean 注入了。

代码演示:

@Lazy
@Component
public class MyBean {

    public MyBean() {
        System.out.println("My bean init success.");
    }

}

1、 使用构造函数注入

@Service
public class MyService {

    private MyBean myBean;

    public MyService(MyBean myBean) {
        this.myBean = myBean;
    }

    public void exec() {
        System.out.println("exec suc");
    }

}

2、 @Resource 注入

@Service
public class MyService {

    @Resource
    private MyBean myBean;

    public void exec() {
        System.out.println("exec suc");
    }

}

3、 @Autowired 注入

@Service
public class MyService {

    private MyBean myBean;

    @Autowired
    public void setMyBean(MyBean myBean) {
        this.myBean = myBean;
    }

    public void exec() {
        myBean.exec();
    }

}

测试结果

失效原因

  这个非常好理解,myService 并没有配置@Lazy,所以在启动的时候会被初始化。由于 myService 依赖 myBean,myBean 就会被注入。所以这意味着 myBean 要能正常被注入,就得被初始化,如果不初始化就会启动失败。这也就是造成 myBean 延迟初始化失效的原因。

解决方法

  解决方法很简单,在依赖到的地方都配置上@Lazy,避免出现被非延迟初始化的 Bean 注入了。

坑 2. 延迟加载失效:Bean 的作用域错误配置

  @Lazy 注解只对单例(Singleton)作用域的 Bean 有效。默认情况下,Spring 的 Bean 作用域是单例,如果将 Bean 的作用域设置为其他作用域(如原型、请求、会话等)的是不起作用的。

代码演示:

  1. 默认不做任何配置。
@Component
public class MyBean {

    public MyBean() {
        System.out.println("My bean init success.");
    }

    public void exec() {
        System.out.println("exec suc");
    }

}

启动结果:

  通过观察启动结果,可以看到 myBean 在启动的时候被初始化了。

  1. 加上@Lazy
@Lazy
@Component
public class MyBean {

    public MyBean() {
        System.out.println("My bean init success.");
    }

    public void exec() {
        System.out.println("exec suc");
    }

}

启动结果:

  通过观察启动结果,可以看到 myBean 并没有初始化,说明@Lazy生效了。

  1. 设置 scope
@Lazy
@Component
@Scope("prototype")
public class MyBean {

    public MyBean() {
        System.out.println("My bean init success.");
    }

    public void exec() {
        System.out.println("exec suc");
    }

}

启动结果:

  这个时候你会发现,貌似这个结果不对呀。上面提到,@Lazy 注解只对单例(Singleton)作用域的 Bean 有效。但是我已经将 Scope 改为 prototype。 按理来应该是这样:

  控制台会输出My bean init success.,然而事实就是没有。那么这是为什么呢?

原因分析

   由于是增加了@Scope("prototype"),发现结果不符合预期,那我们就从它入手。我们先回顾一下 Spring Bean 的作用域相关的知识。当 Spring Bean 作用域为 prototype时,每次获取 Bean 时都会重新创建一个实例。

  换句话说,也就意味着,当的 Bean 作用域为 prototype 时,Bean 在被使用的才会被初始化,并且每个 Bean 都是全新的。

   诶,在使用的时候被初始化,这不就是延迟初始化吗。改下代码测试一下:

去掉@Lazy

@Component
@Scope("prototype")
public class MyBean {

    public MyBean() {
        System.out.println("My bean init success.");
    }

    public void exec() {
        System.out.println("exec suc");
    }

}

启动结果:

  发现和单独配置@Lazy的效果是一样,并没有被初始化。

结论

  当 bean 作用域是 prototype 时,这些 bean 每次在需要时,都会按需实例化和初始化,因此它们本质上是延迟始化的。所以给他们配置@Lazy是没有意义的。

  在上面的案例,出现这样的情况是因为,在启动的时候 myBean 并没有,被其他 Bean 依赖和使用。所以表现出和@Lazy一样的效果。误以为当 Bean 作用域是 prototype 时,@Lazy可以生效。

总结

  由于 spring bean 的默认作用域是:singleton。所以在启动的时候 bean 会被初始化,如果被标记了@Lazy,会延迟初始化,但是如果被非懒加载的 Bean 注入了,@Lazy会失效。并且@Lazy注解只对单例 singleton 作用域的 Bean 有效。

结尾

  如果觉得对你有帮助,可以多多评论,多多点赞哦,也可以到我的主页看看,说不定有你喜欢的文章,也可以随手点个关注哦,谢谢。

  我是不一样的科技宅,每天进步一点点,体验不一样的生活。我们下期见!

标签:初始化,Lazy,Spring,MyBean,Bean,myBean,public,一不小心
From: https://blog.51cto.com/u_11446735/6275105

相关文章

  • Errors:java: 程序包org.springframework.stereotype不存在(解决方法)
    错误截图错误描述java:程序包org.springframework.stereotype不存在解决更改maven项目运行时的编译方式为maven找到File->Settings->Maven->Runner,进行勾选......
  • SpringMVC中多个拦截器的执行顺序
    在SpringMVC中,多个拦截器的执行顺序是由配置文件中拦截器的顺序来决定的。假设我们有3个拦截器:Interceptor1、Interceptor2、Interceptor3,通过配置文件的方式定义拦截器的顺序,例如:<mvc:interceptors><mvc:interceptor><mvc:mappingpath="/**"/><bean......
  • Spring底层原理与源码
    Bean生命周期步骤详情UserService->无参构造方法->普通对象->依赖注入(给对象赋值)->afterPropertiesSet()->初始化后(AOP)->代理对象->Map<beanName,Bean对象>:单例池Bean实例化->JVM实例化Bean初始化->afterPropertiesSet()......
  • 05-面试必会-SpringBoot&SpringCloud
    01-讲一讲SpringBoot自动装配的原理1.在SpringBoot项目的启动引导类上都有一个注解@SpringBootApplication这个注解是一个复合注解,其中有三个注解构成,分别是@SpringBootConfiguration:是@Configuration的派生注解,标注当前类是一个SpringBoot的配置类@......
  • spring遇到的几个错误(5.14)
     先保存配置文件file→ProjectStructure→Modules 详细问题 ......
  • 三、SpringCloud Alibab使用nacos作为配置中心
    nacos不仅作为注册中心,还可以作为配置中心。 在Consumer添加依赖:<dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId></dependency><dependency>......
  • Springcloud 开始来了解
    为什么要学习springcloud?“微服务”一词源于MartinFowler的名为Microservices的博文,简单地说,微服务是系统架构上的一种设计风格,它的主旨是将一个原本独立的系统拆分成多个小型服务,这些小型服务都在各自独立的进程中运行,服务之间通过基于HTTP的RESTfulAPI进行通信协作。......
  • 错问题:--谷粒商城--org.springframework.beans.factory.beancreationexception: error
    做谷粒商城使用人人开源的时候,导入nacos时出现这个问题最后发现是自己在导入时没有把这个版本改掉,最终出现问题,大概原因就是spring的版本和nacos的版本不匹配。<parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent<......
  • spring线程池ThreadPoolTaskExecutor
    1.自定义yml属性配置thread:pool:corePoolSize:5maxPoolSize:10queueCapacity:10keepAliveSeconds:1202.自定义线程池配置类@Configuration//配置类@EnableAsync//开线线程异步支持publicclassMyThreadPoolExecutor{@Autowiredprivat......
  • java基于springboot+vue的房屋租赁管理系统、大学生租房管理系统,附源码+数据库+lw文档
    1、项目介绍根据大学生租房系统的功能需求,进行系统设计。前台功能:进入系统可以实现首页,房屋信息,房屋评价,公告资讯,个人中心,后台管理,意见反馈等功能进行操作;后台主要是管理员,房主和用户,主要功能包括首页,个人中心,房主管理,用户管理,房屋类型管理,房屋信息管理,预约看房管理,定金留房管......