为什么`SpringApplication.run`不能懒惰地创建bean?

问题描述

如果您在项目中使用单例作用域定义了组件类,并且在main()中除了启动应用程序上下文之外不执行任何操作,例如:

ConfigurableApplicationContext context = SpringBootApplication.run(MyMainClass.class,args)

context将已经指向包含这些类的单例实例的ApplicationContext。因此,无论您键入context.getBean(MyContainerClass.class)多少次,您都将得到MyContainerClass的相同实例。

但是该对象的创建可能非常繁重,甚至取决于运行时约束(数据库连接,网络套接字管理,工作)。我知道重载构造函数并不是很好的编程习惯,但是即使在POJO中,也有可能 发生。

我想知道是否存在一种特殊的逻辑来在创建上下文时立即创建单例,而不是第一次调用getBean()。也许从SpringBoot到程序员的合同是这样写的:“如果您用@Component注释了一个类,那么您确实应该在某个时候检索bean:我假设您知道这一点,并且愿意承担构建成本当上下文被提出时。”也许不吧。不确定。

解决方法

实际上,Spring Boot(2.3.3)可以延迟加载@Beans。只需将下面的属性添加到您的application.properties文件中即可。初始化应该延迟进行。

spring.main.lazy-initialization = true

在整个应用程序中启用延迟初始化可能会产生正面和负面影响。请参见Baeldung中描述的惰性初始化的效果:

  1. 惰性初始化可能会减少应用程序启动时创建的bean的数量–因此,我们可以缩短应用程序的启动时间

  2. 由于在需要它们之前都不创建任何Bean,因此我们可以掩盖问题,使其在运行时而不是启动时出现

  3. 问题可能包括内存不足错误,配置错误或发现类定义的错误

  4. 此外,当我们处于Web上下文中时,按需触发Bean创建会增加HTTP请求的延迟– Bean创建只会影响第一个请求,但这可能会对负载产生负面影响-平衡和自动缩放。

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...