Spring內(nèi)核研究-set方法注入和構(gòu)造函數(shù)注入
Spring種提供了2種常用的注入方式,set方法注入和構(gòu)造函數(shù)注入。由于這2種注入方式很相似,都可以滿足我們的需求,所以在大多數(shù)情況下我們忽視了這2種注入方式的區(qū)別。下面讓我們看看這2種注入方式的特點(diǎn)。
我們先看看Spring在使用set方法注入時(shí),是怎樣實(shí)例化一個(gè)Bean和Bean的合作者的:

在A中有一個(gè)setB方法用來接收B對象的實(shí)例。那么Spring實(shí)例化A對象的過程如下:

在不考慮Bean的初始化方法和一些Spring回調(diào)的情況下,Spring首先去調(diào)用A對象的構(gòu)造函數(shù)實(shí)例化A,然后查找A依賴的對象本例子中是B(合作者)。一但找到合作者,Spring就會調(diào)用合作者(B)的構(gòu)造函數(shù)實(shí)例化B。如果B還有依賴的對象Spring會把B上依賴的所有對象都按照相同的機(jī)制實(shí)例化然后調(diào)用A對象的setB(B b)把b對象注入給A。
因?yàn)镾pring調(diào)用一個(gè)對象的set方法注入前,這個(gè)對象必須先被實(shí)例化。所以在"使用set方法注入"的情況下Spring會首先調(diào)用對象的構(gòu)造函數(shù)。
我們在來看通過構(gòu)造函數(shù)注入的過程:

如果發(fā)現(xiàn)配置了對象的構(gòu)造注入,那么Spring會在調(diào)用構(gòu)造函數(shù)前把構(gòu)造函數(shù)需要的依賴對象都實(shí)例化好,然后再把這些實(shí)例化后的對象作為參數(shù)去調(diào)用構(gòu)造函數(shù)。
在使用構(gòu)造函數(shù)和set方法依賴注入時(shí),Spring處理對象和對象依賴的對象的順序時(shí)不一樣的。一般把一個(gè)Bean設(shè)計(jì)為構(gòu)造函數(shù)接收依賴對象時(shí),其實(shí)是表達(dá)了這樣一種關(guān)系:他們(依賴對象)不存在時(shí)我也不存在,即“沒有他們就沒有我”。
通過構(gòu)造函數(shù)的注入方式其實(shí)表達(dá)了2個(gè)對象間的一種強(qiáng)的聚合關(guān)系:組合關(guān)系。就比如一輛車如果沒有輪子、引擎等部件那么車也就不存在了。而且車是由若干重要部件組成的,在這些部件沒有的情況下車也不可能存在。這里車和他的重要部件就時(shí)組合的關(guān)系。如果你的應(yīng)用中有這樣類似的場景那么你應(yīng)該使用“構(gòu)造函數(shù)注入”的方式管理他們的關(guān)系。“構(gòu)造函數(shù)注入”可以保證合作者先創(chuàng)建,在后在創(chuàng)建自己。
通過set方法注入的方式表達(dá)了2個(gè)對象間較弱的依賴關(guān)系:聚合關(guān)系。就像一輛車,如果沒有車內(nèi)音像車也時(shí)可以工作的。當(dāng)你不要求合作者于自己被創(chuàng)建時(shí),“set方法注入”注入比較合適。
雖然在理論上“構(gòu)造函數(shù)注入”和“set方法注入”代表2種不同的依賴強(qiáng)度,但是在spring中,spring并不會把無效的合作者傳遞給一個(gè)bean。如果合作者無效或不存在spring會拋出異常,這樣spring保證一個(gè)對象的合作者都是可用的。所以在spring中,“構(gòu)造函數(shù)注入”和“set方法注入”唯一的區(qū)別在于2種方式創(chuàng)建合作者的順序不同。
使用構(gòu)造函數(shù)依賴注入時(shí),Spring保證所有一個(gè)對象所有依賴的對象先實(shí)例化后,才實(shí)例化這個(gè)對象。(沒有他們就沒有我原則)
使用set方法依賴注入時(shí),Spring首先實(shí)例化對象,然后才實(shí)例化所有依賴的對象。
posted on 2007-04-09 02:56 crazy zerlot 閱讀(1457) 評論(0) 編輯 收藏