開發技巧!@Lazy注解這5種用法非常實用
環境:Spring6.1.8
1. 簡介
Spring中的@Lazy注解主要用于實現惰性加載(延遲加載),它可以應用在類、方法、構造方法、參數和字段上。以下是@Lazy注解的作用、應用場景。
1.1 作用
在Spring框架中,默認情況下,所有的Bean在容器啟動時都會被初始化。但是,有些Bean的初始化可能涉及到一些重量級的操作,如網絡IO操作、復雜計算等,這些操作會消耗大量的系統資源。通過使用@Lazy注解,可以讓這些Bean在真正需要時才進行初始化,從而提高系統的啟動速度和性能。
1.2 應用場景
- 提升系統啟動速度:當應用包含大量的Bean,如果存在某些Bean初始化操作非常耗時(如網絡IO操作或復雜耗時計算),通過@Lazy注解可以顯著提升系統的啟動速度。如:應用啟動時,需要從Redis讀取大量的緩存數據,如果將此Bean使用@Lazy標注,那么應用啟動會非常快,而當在使用緩存服務時才去讀取redis初始化數據。
- 解決循環依賴:如果兩個Bean之間存在循環依賴,即A依賴B,B又依賴A(構造函數注入),這會導致Spring容器在初始化這些Bean時陷入死循環。使用@Lazy注解可以解決這類問題。
- 單例Bean正確注入多例Bean:如果A是單例,B是多例,在A中注入B實例,要想正確的注入(每次使用B時都是新對象)通過使用@Lazy能夠輕松解決。
以上是關于@Lazy注解的簡介及應用場景,接下來將詳細介紹@Lazy的5種使用方式。
2. 實戰案例
2.1 環境準備
public class PersonDAO {
}
public class PersonService {
private PersonDAO dao ;
public String toString() {
return "PersonService [dao=" + dao.getClass() + "]";
}
}
// 測試入口代碼
try (AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext()) {
context.register(PersonDAO.class) ;
context.register(PersonService.class) ;
context.refresh() ;
System.out.println(context.getBean(PersonService.class)) ;
}
接下的每個示例都將基于上面的類進行。
2.2 字段注入
// @Resource
// @Autowired
@Lazy
private PersonDAO dao ;
輸出結果
PersonService [dao=class com.pack.PersonDAO$$SpringCGLIB$$0]
通過@Lazy標注的字段,最終注入的是代理類(不管上面使用的@Resource還是@Autowired)。
注:在上面測試入口代碼中,我們使用的是AnnotationConfigApplicationContext,如果你使用的是GenericApplicationContext那么在默認情況下@Autowired是不會生效的,這時候你還需要做如下設置:
DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory() ;
beanFactory.setAutowireCandidateResolver(new ContextAnnotationAutowireCandidateResolver()) ;
將此BeanFactory設置到ApplicationContext中即可。
2.3 方法注入
@Resource
@Lazy
public void setPersonDAO(PersonDAO dao) {
this.dao = dao ;
}
輸出結果
PersonService [dao=class com.pack.PersonDAO$$SpringCGLIB$$0]
同樣的生成了代理。
方法注入,你還可以將@Lazy放到方法參數上,如下示例:
@Resource
public void setPersonDAO(@Lazy PersonDAO dao) {
this.dao = dao ;
}
這種方式也是會被生成代理對象。
2.4 構造函數注入
@Lazy
public PersonService(PersonDAO dao) {
this.dao = dao ;
}
輸出結果
PersonService [dao=class com.pack.PersonDAO$$SpringCGLIB$$0]
同樣,注解也可以使用在參數上
public PersonService(@Lazy PersonDAO dao) {
this.dao = dao ;
}
構造函數注入與方法注入基本一致。
2.5 單例Bean注入多例Bean
修改PersonDAO;
@Scope("prototype")
public class PersonDAO {
}
通過@Scope將其聲明為多例。
修改PersonService隨意添加一個方法。
public class PersonService {
@Autowired
private PersonDAO dao ;
public void save() {
System.out.printf("PersonDAO hashCode: %s%n", dao) ;
}
}
測試類:
PersonService ps = context.getBean(PersonService.class);
ps.save() ;
ps.save() ;
ps.save() ;
當dao字段上不添加@Lazy注解時,輸出結果:
PersonDAO hashCode: com.pack.PersonDAO@66565121
PersonDAO hashCode: com.pack.PersonDAO@66565121
PersonDAO hashCode: com.pack.PersonDAO@66565121
每次都是同一個對象,這不是我們期望的結果
dao字段添加@Lazy注解后,再次運行
PersonDAO hashCode: com.pack.PersonDAO@73a2e526
PersonDAO hashCode: com.pack.PersonDAO@13f95696
PersonDAO hashCode: com.pack.PersonDAO@68be8808
正確的輸出結果,每次使用都是不同的實例。
2.6 循環依賴
class class A {
private B b ;
public A(B b) {
this.b = b ;
}
}
public class B {
private A a ;
public B(A a) {
this.a = a ;
}
}
上面的依賴通過構造方法注入,這種情況下容器啟動是會報錯的,如下:
圖片
出現循環依賴錯誤,通過@Lazy注解解決此問題,只需要在任意類的構造函數上使用@Lazy注解,如下:
public A(@Lazy B b) {
this.b = b ;
}
只需要在其中一方加入了@Lazy注解后,問題得到解決。