Spring 的注入方式有哪些?該如何選擇?
作為 Java程序員都知道,沒(méi)有依賴注入,Spring 框架就是無(wú)法實(shí)現(xiàn)的,那么,在 Spring 框架中,常見(jiàn)的依賴注入方式有哪些呢?我們?cè)撊绾芜x擇?這篇文章來(lái)聊一聊。
從整體上來(lái)看,Spring 的依賴注入有四種方式:
- 構(gòu)造器注入
- setter 方法注入
- 字段注入
- 接口注入
下面,我們將分別分析它們的原理,以及它們的優(yōu)缺點(diǎn)。
1. 構(gòu)造器注入
構(gòu)造器注入(Constructor Injection)是指通過(guò)構(gòu)造函數(shù)傳入依賴的對(duì)象。Spring 容器在創(chuàng)建 bean 時(shí)會(huì)調(diào)用它的構(gòu)造函數(shù),并將所需的依賴項(xiàng)作為參數(shù)傳入。
如下示例展示如何通過(guò)構(gòu)造器注入 bean。
public class Service {
private final Repository repository;
public MyService(Repository repository) {
this.repository = repository;
}
}
優(yōu)點(diǎn):
- 強(qiáng)制性依賴性:在對(duì)象創(chuàng)建時(shí),所有必要的依賴項(xiàng)都必須提供,減少了在運(yùn)行時(shí)出現(xiàn)空指針的風(fēng)險(xiǎn)。
- 不可變性:被注入的依賴可以聲明為 final,使得一旦初始化后對(duì)象的狀態(tài)不可更改,從而增強(qiáng)了對(duì)象的安全性。
- 便于單元測(cè)試:構(gòu)造器參數(shù)很容易模擬(mock)或替代,方便測(cè)試。
缺點(diǎn):
- 復(fù)雜性:如果一個(gè)類有很多依賴,則構(gòu)造函數(shù)可能變得非常復(fù)雜,導(dǎo)致可讀性差。
- 潛在的構(gòu)造函數(shù)過(guò)載:當(dāng)添加新的依賴時(shí),可能需要重載多個(gè)構(gòu)造函數(shù),增加了維護(hù)成本。
構(gòu)造器注入是工作中比較推薦的一種方式,因?yàn)樗幾g期行為,可以減少空指針。但是,如果一個(gè)類需要很多依賴,構(gòu)造器注入會(huì)導(dǎo)致代碼比較臃腫。
2. Setter 注入
Setter 注入(Setter Injection)是指通過(guò) setter 方法注入依賴的對(duì)象。Spring 在創(chuàng)建 bean 后,通過(guò)調(diào)用 setter 方法來(lái)設(shè)置依賴項(xiàng)。
如下示例展示如何通過(guò) Setter注入 bean。
public class Service {
private Repository repository;
public void setMyRepository(Repository repository) {
this.repository = repository;
}
}
優(yōu)點(diǎn):
- 靈活性:可以選擇性地注入依賴項(xiàng),允許在對(duì)象創(chuàng)建后進(jìn)行注入,適合可選的依賴。
- 清晰的配置:可以通過(guò) setter 方法明確地配置并查看依賴關(guān)系。
缺點(diǎn):
- 非強(qiáng)制性依賴:在對(duì)象創(chuàng)建后如果沒(méi)有設(shè)置必需的依賴,可能導(dǎo)致運(yùn)行時(shí)的空指針異常。
- 可變性:依賴可以在對(duì)象生命周期內(nèi)被更改,可能導(dǎo)致不一致的狀態(tài)。
3. 接口注入
接口注入(Interface Injection)是一個(gè)較少使用的方式,通過(guò)一個(gè)接口將依賴注入到類中。通常實(shí)現(xiàn)這個(gè)接口的類會(huì)提供注入的具體實(shí)現(xiàn)。
如下示例展示如何通過(guò)接口注入 bean。
public interface DependencyInjector {
void inject(Service service);
}
優(yōu)點(diǎn):
- 靈活性:可以根據(jù)不同的實(shí)現(xiàn)提供不同的注入方式。
- 清晰性:接口明確了所需要的依賴,增強(qiáng)了代碼的可理解性。
缺點(diǎn):
- 實(shí)現(xiàn)復(fù)雜性:需要定義額外的接口,增加了復(fù)雜性和維護(hù)成本。
- 使用頻率低:由于實(shí)現(xiàn)復(fù)雜,通常不被廣泛采用。
4. 字段注入
字段注入(Field Injection)是直接將依賴注入到類的字段中,通常使用反射和注解。通常會(huì) 使用 Spring 提供的注解(如 @Autowired, @Inject, @Resource)。這種方式較為簡(jiǎn)潔,但是會(huì)有 NPE的風(fēng)險(xiǎn)。
如下示例展示如何通過(guò)字段注入 bean。
public class Service {
@Autowired
private Repository repository;
}
優(yōu)點(diǎn):
- 簡(jiǎn)潔性:代碼量較少,使用反射和注解可自動(dòng)完成依賴注入,易于理解。
- 方便快捷:無(wú)需編寫構(gòu)造函數(shù)或 setter 方法。
缺點(diǎn):
- 不易于測(cè)試:由于字段是私有的,通常不容易替換依賴項(xiàng)進(jìn)行測(cè)試。
- 依賴性不明顯:依賴關(guān)系并不明確,影響代碼的可讀性和維護(hù)性。
- 不支持不可變性:不能將字段聲明為 final,這可能導(dǎo)致不一致的狀態(tài)。
5. 總結(jié)
本文,我們分析了 Spring的 4種注入方式,在選擇依賴注入方式時(shí),我們應(yīng)該根據(jù)具體情況和項(xiàng)目來(lái)決定。根據(jù)工作經(jīng)驗(yàn),建議如下:
- 構(gòu)造器注入是最推薦的方式,特別是在需要強(qiáng)制性依賴和不可變性的場(chǎng)景下。
- 如果無(wú)法通過(guò)構(gòu)造器注入,再選擇 setter注入。
- 如果上述兩者方式都不適用,字段注入則是最后的選擇,雖然使用簡(jiǎn)單,但是容易產(chǎn)生NPE。