问题描述
如果您在项目中使用单例作用域定义了组件类,并且在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中描述的惰性初始化的效果:
-
惰性初始化可能会减少应用程序启动时创建的bean的数量–因此,我们可以缩短应用程序的启动时间
-
由于在需要它们之前都不创建任何Bean,因此我们可以掩盖问题,使其在运行时而不是启动时出现
-
问题可能包括内存不足错误,配置错误或发现类定义的错误
-
此外,当我们处于Web上下文中时,按需触发Bean创建会增加HTTP请求的延迟– Bean创建只会影响第一个请求,但这可能会对负载产生负面影响-平衡和自动缩放。