Spring依賴注入原理分析
下面談談Spring是如何實現反轉模式IOC/">IOC或依賴注入模式DI:
平時,我們需要生成一個物件,使用new語法,如一個類為A
public class A{ public void myMethod(){ System.out.println("hello"); } }
如果我們在 B中呼叫A,那麼如下程式碼:
public class B{ public void invoke(){ A a = new A(); a.myMethod(); } }
每次執行invoke方法時,都要生成一個A物件,如果A物件程式碼較長,這是費時的事
情。於是有如下寫法:
public class B{ A a = new A(); public void invoke(){ a.myMethod(); } }
將A物件變成B的類屬性。 如果我們不想在B中實現A的例項,也就是不想立即new A(),而是想通過外界傳入, 注意,如果你想知道為什麼,這裡涉及到設計模式以及解耦等因素,進一步感興趣者可學習 本站的GoF 23 種設計模式。
如果想讓A的例項從外界傳入,有兩種寫法:
public class B{ A a; public void setA(A a){ this.a = a; } public A getA(){ return a; } public void invoke(){ a.myMethod(); } }
這種寫法,A並沒有被例項化,需要通過外界呼叫setA方法,將A的物件例項賦入B中. 或者通過B的建構函式傳入,如下:
public class B{ A a; public B(A a){ this.a = a; } public void invoke(){ a.myMethod(); } }
上述兩種寫法在程式設計中是經常發生的,B作為呼叫者,A是被呼叫者,A的例項化不在 呼叫者B內部中完成,而是通過建構函式或setXXX方法賦值進來,這種方式我們稱為依賴 性注射(IoC 模式),B 和A 的依賴聯絡是通過建構函式或setXXX 方法賦值進來,這樣, 最大程度解耦了呼叫者B和被呼叫者A之間的耦合聯絡。
Spring如何實現依賴注射?
上文提到:A的例項化不在呼叫者B內部中完成,而是通過建構函式或setXXX 方法賦 值進來,Spring實際就是完成這個賦值的過程。 為了讓Spring自動完成B程式碼中的A的例項化,需要通過配置檔案告訴Spring有關A 的類的屬性,這個配置是applicationContext.xml檔案。 在 applicationContext.xml中,我們先定義JavaBeans為B的配置:
<beans> <bean id="b" class="springsimple.B"/> </beans>
這是最常用的JavaBeans的定義,id相當於物件名,當前檔案應該是唯一。後來Spring使用@Component替代。
再在applicationContext.xml定義A的配置如下:
<beans> <bean id="b" class="springsimple.B"/> <bean id="a" class="springsimple.A"/> </beans>
這樣我們告訴Spring我們有兩個JavaBeans,現在解決關鍵問題,B程式碼中還呼叫了A, 那麼如何讓Spring將A的例項注射到B中?使用Spring配置的property語法。具體配置如 下:
<beans> <bean id="b" class="springsimple.B"> <property name="a"><ref local="a" /></property> <!— 增加這一行--> </bean> <bean id="a" class="springsimple.A" /> </beans>
增加一行說明:B 的屬性a 指向了a,這樣,Spring 會知道B 中屬性a 的例項就是 springsimple.A,在B例項化時將會將B中的a 實現例項化,這是通過setA方法注射進入。 注意,property name="a"中的a 是setA字元中去掉set 後的字串,這個字串第一個 必須是小寫,例如,如果B中有setOneA方法,那麼,配置檔案應該是property name="oneA"。
在Spring Boot以後版本已經可以使用@Autowire進行自動匹配,無需如此繁瑣配置了。