前言
在Spring配置檔案中使用XML檔案進行配置,實際上是讓Spring執行了相應的代碼,例如:
使用<bean>元素,實際上是讓Spring執行無參或有參構造器
使用<property>元素,實際上是讓Spring執行一次setter方法
但Java程式還可能有其他類型的語句:調用getter方法、調用普通方法、通路類或對象的Field等,而Spring也為這種語句提供了對應的配置文法:
調用getter方法:使用PropertyPathFactoryBean
調用類或對象的Filed值:使用FiledRetrievingFactoryBean
調用普通方法:使用MethodInvokingFactoryBean
注入其他Bean的屬性值
PropertyPathFactoryBean用來獲得目标Bean的屬性值(實際上就是調用getter方法傳回的值),獲得的值可以注入給其他的Bean,也可以直接定義新的Bean。看如下的配置檔案:
其中Person類和Son類的屬性可以從配置檔案中看出,這不再給出。主程式如下:
輸出結果:
Bean執行個體的屬性值,不僅可以注入另一個Bean,還可将Bean執行個體的屬性值直接定義成Bean執行個體,這也是通過PropertyPathFactoryBean完成的。對上面的配置檔案增加這樣一段:
執行上面的Test類,把son2換成son1,結果一樣。
注入其他Bean的Field值
通過FieldRetrievingFactoryBean類,可以将其他Bean的Field值注入給其他Bean,或者直接定義新的Bean。下面是配置片段:
測試主程式與上文定義的類似,這裡不再提供,執行結果如下:
在這個配置中,son對象的age的值,等于java.sql.Connection.TRANSACTION_SERIALIZABLE的值。在上面的 定義中,定義FieldRetrievingFactoryBean工廠Bean時,指定的id并不是該Bean執行個體的唯一辨別,而是指定Field的表 達式(即将要被取出來的值)。
注意:Field既可以是靜态的,也可以是非靜态的。上面的配置片段指定的Field表達式是靜态Field值,是以可以通過類名直接通路。如果Field值是非靜态的,則應該通過容器中已經存在的Bean來通路——即Field表達式的第一個短語應該是容器中已經存在的Bean。
Field值也可以定義成Bean執行個體,例如,在配置檔案中增加下面一段:
在主程式中增加如下輸出:
執行結果和上文一樣。
使用FieldRetrievingFactoryBean擷取Field值時,必須指定如下兩個屬性:
targetClass或targetObject:分别用于指定Field值所在的目标類或目标對象。如果需要獲得的Field是靜态的,則使用targetClass指定目标類;如果Field是非靜态的,則使用targetObject指定目标對象
targetField:指定目标類或目标對象的Field名
如果Field是個靜态Field,則有一種更加簡潔的寫法:
注入其他Bean的方法傳回值
通過MethodInvokingFactoryBean工廠Bean,可将目标方法的傳回值注入為Bean的屬性值。這個工廠Bean用 來擷取指定方法的傳回值,該方法既可以是靜态方法,也可以是執行個體方法;這個值既可以被注入到指定Bean執行個體的指定屬性,也可以直接定義成Bean執行個體。 看例子:
下面是ValueGenerator:
測試程式依舊列印son1中age的值,代碼略,結果如下:
如果要調用靜态方法,則把配置修改為:
測試結果為:
由于Java是支援重載的,隻給定方法名,還不足以能夠确定調用哪個方法,通過上面的配置能調用成功是因為ValueGenerator中的兩個方法都沒有參數。如果方法中有參數,該如何配置呢?在配置檔案中加入以下内容:
上例中相當于用"java.version"作為參數調用了java.lang.System的getProperty方法,傳回值将建立一個名為javaVersion的Bean。即相當于:
和前文中的Field一樣,如果要調用的方法為靜态方法,也有一種更加簡潔的方法: