您好,登錄后才能下訂單哦!
本篇內(nèi)容介紹了“Spring依賴注入的方式及優(yōu)缺點是什么”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!
在 Spring 中實現(xiàn)依賴注入的常見方式有以下 3 種:
屬性注入(Field Injection);
Setter 注入(Setter Injection);
構(gòu)造方法注入(Constructor Injection)。
它們的具體使用和優(yōu)缺點分析如下。
屬性注入是我們最熟悉,也是日常開發(fā)中使用最多的一種注入方式,它的實現(xiàn)代碼如下:
@RestController public class UserController { // 屬性對象 @Autowired private UserService userService; @RequestMapping("/add") public UserInfo add(String username, String password) { return userService.add(username, password); } }
屬性注入最大的優(yōu)點就是實現(xiàn)簡單、使用簡單,只需要給變量上添加一個注解(@Autowired),就可以在不 new 對象的情況下,直接獲得注入的對象了(這就是 DI 的功能和魅力所在),所以它的優(yōu)點就是使用簡單。
然而,屬性注入雖然使用簡單,但也存在著很多問題,甚至編譯器 Idea 都會提醒你“不建議使用此注入方式”,Idea 的提示信息如下:
屬性注入的缺點主要包含以下 3 個:
功能性問題:無法注入一個不可變的對象(final 修飾的對象);
通用性問題:只能適應(yīng)于 IoC 容器;
設(shè)計原則問題:更容易違背單一設(shè)計原則。
接下來我們一一來看。
缺點1:功能性問題
使用屬性注入無法注入一個不可變的對象(final 修飾的對象),如下圖所示:
原因也很簡單:在 Java 中 final 對象(不可變)要么直接賦值,要么在構(gòu)造方法中賦值,所以當(dāng)使用屬性注入 final 對象時,它不符合 Java 中 final 的使用規(guī)范,所以就不能注入成功了。
PS:如果要注入一個不可變的對象,要怎么實現(xiàn)呢?使用下面的構(gòu)造方法注入即可。
缺點2:通用性問題
使用屬性注入的方式只適用于 IoC 框架(容器),如果將屬性注入的代碼移植到其他非 IoC 的框架中,那么代碼就無效了,所以屬性注入的通用性不是很好。
缺點3:設(shè)計原則問題
使用屬性注入的方式,因為使用起來很簡單,所以開發(fā)者很容易在一個類中同時注入多個對象,而這些對象的注入是否有必要?是否符合程序設(shè)計中的單一職責(zé)原則?就變成了一個問題。
但可以肯定的是,注入實現(xiàn)越簡單,那么濫用它的概率也越大,所以出現(xiàn)違背單一職責(zé)原則的概率也越大。
注意:這里強調(diào)的是違背設(shè)計原則(單一職責(zé))的可能性,而不是一定會違背設(shè)計原則,二者有著本質(zhì)的區(qū)別。
Setter 注入的實現(xiàn)代碼如下:
@RestController public class UserController { // Setter 注入 private UserService userService; @Autowired public void setUserService(UserService userService) { this.userService = userService; } @RequestMapping("/add") public UserInfo add(String username, String password) { return userService.add(username, password); } }
從上面代碼可以看出,Setter 注入比屬性注入要麻煩很多。
要說 Setter 注入有什么優(yōu)點的話,那么首當(dāng)其沖的就是它完全符合單一職責(zé)的設(shè)計原則,因為每一個 Setter 只針對一個對象。
但它的缺點也很明顯,它的缺點主要體現(xiàn)在以下 2 點:
不能注入不可變對象(final 修飾的對象);
注入的對象可被修改。
接下來我們一一來看。
缺點1:不能注入不可變對象
使用 Setter 注入依然不能注入不可變對象,比如以下注入會報錯:
缺點2:注入對象可被修改
Setter 注入提供了 setXXX 的方法,意味著你可以在任何時候、在任何地方,通過調(diào)用 setXXX 的方法來改變注入對象,所以 Setter 注入的問題是,被注入的對象可能隨時被修改。
構(gòu)造方法注入是 Spring 官方從 4.x 之后推薦的注入方式,它的實現(xiàn)代碼如下:
@RestController public class UserController { // 構(gòu)造方法注入 private UserService userService; @Autowired public UserController(UserService userService) { this.userService = userService; } @RequestMapping("/add") public UserInfo add(String username, String password) { return userService.add(username, password); } }
當(dāng)然,如果當(dāng)前的類中只有一個構(gòu)造方法,那么 @Autowired 也可以省略,所以以上代碼還可以這樣寫:
@RestController public class UserController { // 構(gòu)造方法注入 private UserService userService; public UserController(UserService userService) { this.userService = userService; } @RequestMapping("/add") public UserInfo add(String username, String password) { return userService.add(username, password); } }
構(gòu)造方法注入相比于前兩種注入方法,它可以注入不可變對象,并且它只會執(zhí)行一次,也不存在像 Setter 注入那樣,被注入的對象隨時被修改的情況,它的優(yōu)點有以下 4 個:
可注入不可變對象;
注入對象不會被修改;
注入對象會被完全初始化;
通用性更好。
接下來我們一一來看。
優(yōu)點1:注入不可變對象
使用構(gòu)造方法注入可以注入不可變對象,如下代碼所示:
優(yōu)點2:注入對象不會被修改
構(gòu)造方法注入不會像 Setter 注入那樣,構(gòu)造方法在對象創(chuàng)建時只會執(zhí)行一次,因此它不存在注入對象被隨時(調(diào)用)修改的情況。
優(yōu)點3:完全初始化
因為依賴對象是在構(gòu)造方法中執(zhí)行的,而構(gòu)造方法是在對象創(chuàng)建之初執(zhí)行的,因此被注入的對象在使用之前,會被完全初始化,這也是構(gòu)造方法注入的優(yōu)點之一。
優(yōu)點4:通用性更好
構(gòu)造方法和屬性注入不同,構(gòu)造方法注入可適用于任何環(huán)境,無論是 IoC 框架還是非 IoC 框架,構(gòu)造方法注入的代碼都是通用的,所以它的通用性更好。
“Spring依賴注入的方式及優(yōu)缺點是什么”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。