詳解Spring?延遲初始化遇到的問題
List<坑> 坑列表 = new ArrayList<>(2);
首先,讓我們回顧一下 Spring 延遲初始化的概念。在 Spring 中,延遲初始化指的是將 Bean 的實例化推遲到第一次被使用時,而不是在應(yīng)用程序啟動時就立即創(chuàng)建所有的 Bean。這種延遲加載的機制可以提高應(yīng)用程序的性能和資源利用率。
坑 1. 延遲加載失效,被非延遲初始化的 Bean 注入了。
代碼演示:
@Lazy
@Component
public class MyBean {
public MyBean() {
System.out.println("My bean init success.");
}
}
1、 使用構(gòu)造函數(shù)注入
@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();
}
}
測試結(jié)果

失效原因
這個非常好理解,myService 并沒有配置@Lazy,所以在啟動的時候會被初始化。由于 myService 依賴 myBean,myBean 就會被注入。所以這意味著 myBean 要能正常被注入,就得被初始化,如果不初始化就會啟動失敗。這也就是造成 myBean 延遲初始化失效的原因。
解決方法
解決方法很簡單,在依賴到的地方都配置上@Lazy,避免出現(xiàn)被非延遲初始化的 Bean 注入了。
坑 2. 延遲加載失效:Bean 的作用域錯誤配置
@Lazy 注解只對單例(Singleton)作用域的 Bean 有效。默認情況下,Spring 的 Bean 作用域是單例,如果將 Bean 的作用域設(shè)置為其他作用域(如原型、請求、會話等)的是不起作用的。
代碼演示:
- 默認不做任何配置。
@Component
public class MyBean {
public MyBean() {
System.out.println("My bean init success.");
}
public void exec() {
System.out.println("exec suc");
}
}
啟動結(jié)果:

通過觀察啟動結(jié)果,可以看到 myBean 在啟動的時候被初始化了。
- 加上
@Lazy
@Lazy
@Component
public class MyBean {
public MyBean() {
System.out.println("My bean init success.");
}
public void exec() {
System.out.println("exec suc");
}
}
啟動結(jié)果:

通過觀察啟動結(jié)果,可以看到 myBean 并沒有初始化,說明@Lazy生效了。
- 設(shè)置 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");
}
}
啟動結(jié)果:

這個時候你會發(fā)現(xiàn),貌似這個結(jié)果不對呀。上面提到,@Lazy 注解只對單例(Singleton)作用域的 Bean 有效。但是我已經(jīng)將 Scope 改為 prototype。 按理來應(yīng)該是這樣:

控制臺會輸出My bean init success.,然而事實就是沒有。那么這是為什么呢?
原因分析
由于是增加了@Scope("prototype"),發(fā)現(xiàn)結(jié)果不符合預期,那我們就從它入手。我們先回顧一下 Spring Bean 的作用域相關(guān)的知識。當 Spring Bean 作用域為 prototype時,每次獲取 Bean 時都會重新創(chuàng)建一個實例。
換句話說,也就意味著,當?shù)?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");
}
}
啟動結(jié)果:

發(fā)現(xiàn)和單獨配置@Lazy的效果是一樣,并沒有被初始化。
結(jié)論
當 bean 作用域是 prototype 時,這些 bean 每次在需要時,都會按需實例化和初始化,因此它們本質(zhì)上是延遲始化的。所以給他們配置@Lazy是沒有意義的。
在上面的案例,出現(xiàn)這樣的情況是因為,在啟動的時候 myBean 并沒有,被其他 Bean 依賴和使用。所以表現(xiàn)出和@Lazy一樣的效果。誤以為當 Bean 作用域是 prototype 時,@Lazy可以生效。
總結(jié)
由于 spring bean 的默認作用域是:singleton。所以在啟動的時候 bean 會被初始化,如果被標記了@Lazy,會延遲初始化,但是如果被非懶加載的 Bean 注入了,@Lazy會失效。并且@Lazy注解只對單例 singleton 作用域的 Bean 有效。
結(jié)尾
如果覺得對你有幫助,可以多多評論,多多點贊哦,也可以到我的主頁看看,說不定有你喜歡的文章,也可以隨手點個關(guān)注哦,謝謝。
以上就是詳解Spring 延遲初始化遇到的問題的詳細內(nèi)容,更多關(guān)于Spring 延遲初始化的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
maven項目pom.xml中parent標簽的使用小結(jié)
使用maven是為了更好的幫項目管理包依賴,maven的核心就是pom.xml,當我們需要引入一個jar包時,在pom文件中加上就可以從倉庫中依賴到相應(yīng)的jar包,本文就來介紹一下maven項目pom.xml中parent標簽的使用小結(jié),感興趣的可以了解一下2023-12-12
RabbitMQ消息隊列實現(xiàn)延遲任務(wù)示例
這篇文章主要為大家介紹了RabbitMQ消息隊列實現(xiàn)延遲任務(wù)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步早日升職加薪2022-04-04
Springboot+Flowable?快速實現(xiàn)工作流的開發(fā)流程
這篇文章主要介紹了Springboot+Flowable?快速實現(xiàn)工作流的開發(fā)流程,本文通過實例代碼圖文相結(jié)合給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-02-02
mybatis resultmap 如何為對象賦值的調(diào)用順序
這篇文章主要介紹了mybatis resultmap 如何為對象賦值的調(diào)用順序,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-01-01

