在 spring 容器中,兩個(gè) bean 之間除了通過 <ref> 建立依賴關(guān)系外,還存在著一些特殊關(guān)系。
1 繼承
在面向?qū)ο蟮木幊淘碇校?dāng)多個(gè)類擁有相同的方法和屬性時(shí),則可以引入父類用于消除重復(fù)的代碼 。 而在 spring 容器中,如果多個(gè) bean 存在相同的配置信息,我們可以定義一個(gè)父 bean ,這樣子 bean 將會(huì)自動(dòng)繼承父 bean 的配置信息 。
1
2
3
4
5
6
7
8
9
|
<!-- 父 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 的功能是簡(jiǎn)化子 bean 的配置,所以設(shè)置為抽象類(abstract="true");如果這里沒有把父 bean 設(shè)置為抽象類,那么 spring 容器會(huì)實(shí)例化父 bean 。
2 前置依賴
一般情況下,使用 <ref> 來建立 bean 之間的依賴關(guān)系, spring 容器負(fù)責(zé)管理這些關(guān)系,當(dāng)實(shí)例化一個(gè) bean 時(shí),容器保證該 bean 所依賴的 bean 都已經(jīng)完成了初始化工作。
但在某些情況下,bean 之間的依賴關(guān)系并沒有那么明顯。
假設(shè)這樣一種場(chǎng)景,某系統(tǒng)設(shè)置了一些系統(tǒng)參數(shù)(如密碼有效期、是否開啟監(jiān)控等),這些啟動(dòng)參數(shù)用來控制系統(tǒng)的運(yùn)行邏輯,我們使用一個(gè) setting 類來表示這些參數(shù):
1
2
3
4
5
6
7
8
9
10
|
public class settings { /** * 密碼過期時(shí)間(單位:天) */ public static int pass_timeout = 30 ; /** * 是否開啟監(jiān)控 */ public static boolean is_monitor = false ; } |
在此,我們?yōu)檫@些參數(shù)設(shè)置了默認(rèn)值。系統(tǒng)還有一個(gè)管理后臺(tái),管理員可以通過這個(gè)后臺(tái)調(diào)整這些系統(tǒng)參數(shù)并保存到數(shù)據(jù)庫(kù)中。所以應(yīng)用啟動(dòng)時(shí),需要從數(shù)據(jù)庫(kù)中加載這些系統(tǒng)參數(shù):
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public class system { public system() { init(); } /** * 初始化 */ private void init() { //假設(shè)這些值來源于數(shù)據(jù)庫(kù) settings.pass_timeout = 20 ; settings.is_monitor = true ; } } |
系統(tǒng)有一個(gè)密碼過期管理器,它會(huì)根據(jù)系統(tǒng)參數(shù)中的【密碼過期的天數(shù)】,來創(chuàng)建檢測(cè)密碼是否過期的定時(shí)任務(wù):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public class passmanager { int timeout; public passmanager() { timeout = settings.pass_timeout; timertask(); } /** * 檢測(cè)密碼是否過期的定時(shí)任務(wù) */ private void timertask() { } public int gettimeout() { return timeout; } } |
雖然 passmanager 并沒有直接依賴于 settings,但從邏輯上來看,passmanager 希望 system 加載初始化系統(tǒng)參數(shù)后再啟動(dòng)。
spring 中可以通過 depends-on 屬性顯式地指定 bean 的前置依賴 bean, 保證這個(gè) bean 在實(shí)例化之前,它的前置依賴 bean 已經(jīng)加載完畢。
1
2
3
|
<bean id= "system" class = "net.deniro.spring4.bean.system" /> <bean id= "manager" class = "net.deniro.spring4.bean.passmanager" depends-on= "system" /> |
如果前置依賴于多個(gè) bean ,那么可以通過逗號(hào)、空格或分號(hào)的方式來配置 bean 名稱 。
3 引用 id
假設(shè)一個(gè) bean 需要引用另一個(gè) bean 的 id 值(beanname),這一般用于在運(yùn)行期間在 bean 中通過 getbean(beanname) 方法獲取另一個(gè) bean 的情境。
可以這樣配置:
1
2
3
|
<bean id= "author" class = "net.deniro.spring4.bean.author" /> <bean id= "book" class = "net.deniro.spring4.bean.book" p:authorid= "author" /> |
book 中新增 authorid 屬性:
1
2
3
4
|
/** * author bean 的 id */ private string authorid; |
雖然可以以這種字面值的形式進(jìn)行設(shè)置,但兩者之間并沒有建立真正的引用關(guān)系。所以只有等到具體調(diào)用時(shí)才會(huì)發(fā)現(xiàn)配置錯(cuò)誤。
spring 提供了 <idref> 元素標(biāo)簽,通過 <idref> 引用另一個(gè) bean 的名稱,這樣在容器啟動(dòng)時(shí),就會(huì)檢查引用關(guān)系的正確性,可以提前發(fā)現(xiàn)錯(cuò)誤的配置信息。
1
2
3
4
5
6
7
|
<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ā)生錯(cuò)誤,spring 容器啟動(dòng)時(shí)就會(huì)拋出 beandefinitionstoreexception,而且 ide 的xml 分析器也會(huì)提前發(fā)現(xiàn)引用錯(cuò)誤,所以推薦使用 <idref> 元素標(biāo)簽的方式來引用 id。
總結(jié)
以上所述是小編給大家介紹的spring bean 之間的特殊關(guān)系,希望對(duì)大家有所幫助,如果大家有任何疑問歡迎給我留言,小編會(huì)及時(shí)回復(fù)大家的!
原文鏈接:https://www.jianshu.com/p/e283906af3a9