詳解Spring Bean 之間的特殊關(guān)系
在 Spring 容器中,兩個 Bean 之間除了通過 <ref> 建立依賴關(guān)系外,還存在著一些特殊關(guān)系。
1 繼承
在面向?qū)ο蟮木幊淘碇?,?dāng)多個類擁有相同的方法和屬性時,則可以引入父類用于消除重復(fù)的代碼 。 而在 Spring 容器中,如果多個 Bean 存在相同的配置信息,我們可以定義一個父 Bean ,這樣子 Bean 將會自動繼承父 Bean 的配置信息 。
<!-- 父 Bean--> <bean id="abstractBook" class="net.deniro.spring4.bean.Book" p:name="面紗" abstract="true"> </bean> <!-- 子 Bean--> <bean id="book1" class="net.deniro.spring4.bean.Book" p:press="重慶出版社" parent="abstractBook"/> <bean id="book2" class="net.deniro.spring4.bean.Book" p:press="上海譯文出版社" parent="abstractBook"/>
一般情況下,父 Bean 的功能是簡化子 Bean 的配置,所以設(shè)置為抽象類(abstract="true");如果這里沒有把父 Bean 設(shè)置為抽象類,那么 Spring 容器會實(shí)例化父 Bean 。
2 前置依賴
一般情況下,使用 <ref> 來建立 Bean 之間的依賴關(guān)系, Spring 容器負(fù)責(zé)管理這些關(guān)系,當(dāng)實(shí)例化一個 Bean 時,容器保證該 Bean 所依賴的 Bean 都已經(jīng)完成了初始化工作。
但在某些情況下,Bean 之間的依賴關(guān)系并沒有那么明顯。
假設(shè)這樣一種場景,某系統(tǒng)設(shè)置了一些系統(tǒng)參數(shù)(如密碼有效期、是否開啟監(jiān)控等),這些啟動參數(shù)用來控制系統(tǒng)的運(yùn)行邏輯,我們使用一個 Setting 類來表示這些參數(shù):
public class Settings {
/**
* 密碼過期時間(單位:天)
*/
public static int PASS_TIMEOUT = 30;
/**
* 是否開啟監(jiān)控
*/
public static boolean IS_MONITOR = false;
}
在此,我們?yōu)檫@些參數(shù)設(shè)置了默認(rèn)值。系統(tǒng)還有一個管理后臺,管理員可以通過這個后臺調(diào)整這些系統(tǒng)參數(shù)并保存到數(shù)據(jù)庫中。所以應(yīng)用啟動時,需要從數(shù)據(jù)庫中加載這些系統(tǒng)參數(shù):
public class System {
public System() {
init();
}
/**
* 初始化
*/
private void init() {
//假設(shè)這些值來源于數(shù)據(jù)庫
Settings.PASS_TIMEOUT = 20;
Settings.IS_MONITOR = true;
}
}
系統(tǒng)有一個密碼過期管理器,它會根據(jù)系統(tǒng)參數(shù)中的【密碼過期的天數(shù)】,來創(chuàng)建檢測密碼是否過期的定時任務(wù):
public class PassManager {
int timeout;
public PassManager() {
timeout = Settings.PASS_TIMEOUT;
timerTask();
}
/**
* 檢測密碼是否過期的定時任務(wù)
*/
private void timerTask() {
}
public int getTimeout() {
return timeout;
}
}
雖然 PassManager 并沒有直接依賴于 Settings,但從邏輯上來看,PassManager 希望 System 加載初始化系統(tǒng)參數(shù)后再啟動。
Spring 中可以通過 depends-on 屬性顯式地指定 Bean 的前置依賴 Bean, 保證這個 Bean 在實(shí)例化之前,它的前置依賴 Bean 已經(jīng)加載完畢。
<bean id="system" class="net.deniro.spring4.bean.System"/> <bean id="manager" class="net.deniro.spring4.bean.PassManager" depends-on="system"/>
如果前置依賴于多個 Bean ,那么可以通過逗號、空格或分號的方式來配置 Bean 名稱 。
3 引用 ID
假設(shè)一個 Bean 需要引用另一個 Bean 的 id 值(beanName),這一般用于在運(yùn)行期間在 Bean 中通過 getBean(beanName) 方法獲取另一個 Bean 的情境。
可以這樣配置:
<bean id="author" class="net.deniro.spring4.bean.Author"/> <bean id="book" class="net.deniro.spring4.bean.Book" p:authorId="author"/>
Book 中新增 authorId 屬性:
/** * author Bean 的 ID */ private String authorId;
雖然可以以這種字面值的形式進(jìn)行設(shè)置,但兩者之間并沒有建立真正的引用關(guān)系。所以只有等到具體調(diào)用時才會發(fā)現(xiàn)配置錯誤。
Spring 提供了 <idref> 元素標(biāo)簽,通過 <idref> 引用另一個 Bean 的名稱,這樣在容器啟動時,就會檢查引用關(guān)系的正確性,可以提前發(fā)現(xiàn)錯誤的配置信息。
<bean id="author10" class="net.deniro.spring4.bean.Author"/>
<bean id="book10" class="net.deniro.spring4.bean.Book"
>
<property name="authorId">
<idref bean="author10"/>
</property>
</bean>
如果配置發(fā)生錯誤,Spring 容器啟動時就會拋出 BeanDefinitionStoreException,而且 IDE 的XML 分析器也會提前發(fā)現(xiàn)引用錯誤,所以推薦使用 <idref> 元素標(biāo)簽的方式來引用 ID。
總結(jié)
以上所述是小編給大家介紹的Spring Bean 之間的特殊關(guān)系,希望對大家有所幫助,如果大家有任何疑問歡迎給我留言,小編會及時回復(fù)大家的!
相關(guān)文章
java項(xiàng)目導(dǎo)出為.exe執(zhí)行文件的方法步驟
最近做了個項(xiàng)目,想要轉(zhuǎn)換成可執(zhí)行文件,那么java項(xiàng)目如何導(dǎo)出為.exe執(zhí)行文件,本文就介紹一下,主要使用jar2exe軟件,感興趣的可以了解一下2021-05-05
java?stream使用指南之sorted使用及進(jìn)階方式
這篇文章主要介紹了java?stream使用指南之sorted使用及進(jìn)階方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-05-05
JDBC中Statement和Preparement的使用講解
今天小編就為大家分享一篇關(guān)于JDBC中Statement和Preparement的使用講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-01-01
Spring?Boot項(xiàng)目完美大一統(tǒng)(結(jié)果異常日志統(tǒng)一)
這篇文章主要為大家介紹了Spring?Boot項(xiàng)目完美大一統(tǒng)(結(jié)果異常日志統(tǒng)一)的實(shí)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01
spring項(xiàng)目實(shí)現(xiàn)單元測試過程解析
這篇文章主要介紹了spring項(xiàng)目實(shí)現(xiàn)單元測試過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
Maven 多模塊父子工程的實(shí)現(xiàn)(含Spring Boot示例)
這篇文章主要介紹了Maven 多模塊父子工程的實(shí)現(xiàn)(含Spring Boot示例),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04

