在IoC容器中裝配Bean(基於XML配置)
一、Bean的命名
一般情況下,必須為Bean指定id,id必須唯一不能重複,如果使用name屬性,可以重複。也可以使用匿名的bean。強烈建議通過id指定Bean。
<bean name="#car1,123,$car" class="com.smart.simple.Car"/>
如果同一個Bean使用了相同的name,那麼getBean(beanName)獲取Bean時,會返回後面宣告的那個Bean。
如果id和name都沒有指定,只指定了全類名如:
<bean class="com.smart.simple.Car"/>
<bean class="com.smart.simple.Car"/>
<bean class="com.smart.simple.Car"/>
第一個Bean可以通過getBean(“com.smart.simple.Car”)獲得,第二個Bean通過getBean(“com.smart.simple.Car#1”)獲得。第三個Bean通過getBean(“com.smart.simple.Car#2”)獲得。
二、依賴注入
2.1 屬性注入
屬性注入通過setXxx()方法注入Bean的屬性值或依賴物件,需要注意的是,Bean中不一定有xxx屬性,只是需要提供setter方法。變數的前兩個字母要麼全部大寫,要麼全部小寫。程式碼略。。
2.2 建構函式注入
2.2.1 按型別匹配入參
public class Car{
private String brand;
private double price
public Car(String brand, double price) {
this.brand = brand;
this.price = price;
}
}
<bean id="car1" class="com.smart.simple.Car">
<contructor-arg type="java.lang.String" value="紅旗CA72"/>
<contructor-arg type="double" value="20000"/>
</bean>
2.2.2 按索引匹配入參
public Car(String brand,String corp,double price){
this.brand=brand;
this.corp=corp;
this.price=price;
}
<bean id="car2" class="com.smart.simple.Car">
<contructor-arg index="0" value="紅旗CA72"/>
<contructor-arg index="1" value="中國一汽"/>
<contructor-arg index="2" value="20000"/>
</bean>
2.2.3 聯合使用型別和索引匹配入參
public Car(String brand,String corp,double price){
...
}
public Car(String brand,String corp,int maxSpeed){
...
}
<bean id="car3" class="com.smart.simple.Car">
<contructor-arg type="java.lang.String" index="0" value="紅旗CA72"/>
<contructor-arg type="java.lang.String" index="1" value="中國一汽"/>
<contructor-arg type="int" index="2" value="20000"/>
</bean>
2.2.4 通過自身型別反射入參
public Boss(String name,Car car,Office office){
...
}
<bean id="boss" class="com.smart.simple.Boss">
<contructor-arg value="John"/>
<conturctor-arg ref="car"/>
<contructor-arg ref="office"/>
</bean>
2.2.5 迴圈依賴問題
由於兩個bean都採用建構函式注入,且都通過建構函式入參引用對方,就會發生迴圈依賴問題。如果存在迴圈依賴問題,Spring容器將無法啟動,解決辦法是將狗仔函式注入改為屬性注入就可以了。
public Car(String brand,Boss boss){
}
public Boss(String name,Car car){
}
<bean id="car4" class="com.smart.simple.Car">
<contructor-arg index="0" value="紅旗CA72"/>
<contructor-arg index="1" ref="boss"/>
</bean>
<bean id="boss" class="com.smart.simple.Boss">
<contructor-arg index="0" value="Jone"/>
<contructor-arg index="1" ref="car"/>
</bean>
2.3 工廠方法注入
2.3.1 非靜態工廠方法
public class CarFactory{
public Car createHongQiCar(){
Car car=new Car();
car.setBrand("紅旗CA72");
return car;
}
}
<bean id="carFactory" class="com.smart.factory.CarFacotry"/>
<bean id="car5" factory-bean="carFactory" factory-method="createHongQiCar" />
2.3.2 靜態工廠方法
<bean id="car6" class="com.smart.factory.CarFactory" factory-method="createCar" />
//class指工廠類,factory-method指工廠類方法
三、注入引數詳解
3.1 字面值
當在xml配置檔案中使用(< > & ” ‘)五種符號時,要注意特殊處理:其一使用<![CDATA[紅旗&CA72]]>
特殊標籤,其二是使用xml轉義序列表示這些特殊字元:紅旗&72
。
3.2 引用其他Bean
public class Boss{
public void setCar(Car car){
this.car=car;
}
}
<bean id="car" class="com.smart.simple.Car"/>
<bean id="boss" class="com.smart.simple.Boss">
<property>
<ref bean="car"></ref>
</property>
</bean>
/*
ref元素通過以下3個屬性引用容器中的其他bean:
bean:通過該屬性可以引用統一容器或父容器中的bean,這是最常見的形式
local:只能引用同一配置檔案中定義的bean。
parent:引用父容器中的bean
*/
3.3 內部bean
如果一個Bean只想被一個Bean引用,二不被容器中任何其他的Bean引用,則可以將Bean以內部Bean的方式注入。
<bean id="boss" class="com.smart.simple.Boss">
<property name="car">
<bean class="com.smart.simple.Car" p:maxSpeed="200" p:price="20000"/>
</property>
</bean>
3.4 null值
<property name="brand"><null/></property>
3.5 級聯屬性
<bean id="boss3" class="com.smart.simple.Boss" >
<property name="car.brand" value="吉利CT40"/>
</bean>
//注意car非空
3.6 集合型別屬性
3.6.1 List
public class Boss{
public List favorites=new ArrayList();
public void setFavorites(List favorites){
this.favorites=favorites;
}
}
<bean id="boss1" class="com.smart.simple.Boss">
<property name="favorites">
<list>
<value>看報</value>
<value>游泳</value>
<value>打球</value>
</list>
</property>
</bean>
3.6.2 Map
private Map jobs=new HashMap();
public void setJobs(Map jobs){
this.jobs=jobs;
}
<bean id="boss1" class="com.smart.simple.Boss">
<property name="jobs">
<map>
<entry key="AM" value="會見客戶" />
<entry key="PM" value="內部會議" />
</map>
</property>
</boss>
3.6.3 Set
<bean id="boss1" class="com.smart.simple.Boss">
<property name="favorites">
<set>
<value>看報</value>
<value>游泳</value>
<value>打球</value>
</set>
</property>
</bean>
3.6.4 Properties
private Properties mails=new Properties();
public void setMails(){}
<bean id="boss1" class="com.smart.simple.Boss">
<property name="mails">
<props>
<prop key="jobMail">[email protected]</prop>
<prop key="lifeMail">[email protected]</prop>
</props>
</property>
</bean>
3.6.5 強型別集合
與非強型別集合配置相同。
3.6.6 集合合併
<bean id="boss1" class="com.smart.simple.Boss" abstract="true"> <!--父bean-->
<property name="favorites">
<set>
<value>看報</value>
<value>游泳</value>
<value>打球</value>
</set>
</property>
</bean>
<bean id="childBoss" parent="boss1"> <!--parent 指定父bean-->
<property name="favorites">
<set merge="true"> <!--和父bean中的同名集合屬性合併-->
<value>爬山</value>
<value>逛街</value>
</set>
</property>
</bean>
3.6.7 通過util名稱空間配置集合型別的bean
引入util的名稱空間宣告。
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.3.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-4.3.xsd">
<util:list id="favoritesList" list-class="java.util.LinkedList">
<value>看報</value>
<value>賽車</value>
<value>高爾夫</value>
</util:list>
<util:set id="favoriteSet">
<value>看報</value>
<value>賽車</value>
<value>高爾夫</value>
</util:set>
<util:map id="email">
<entry key="AM" value="會見客戶"></entry>
<entry key="PM" value="內部會議"></entry>
</util:map>
</beans>
四、方法注入
用處:向singleton的Bean中注入prototype的Bean,一種方法是讓singleton實現BeanFactoryAware介面。下面是其餘兩種方法。
4.1 lookup方法注入
public interface MaginBoss{
//通過getBean()返回prototype的Bean,每次都返回新例項
Car getCar
}
<!--需要匯入cglib包-->
<!--prototype例項的bean-->
<bean id="car" class="com.smart.simple.Car" p:brand="紅旗" p:price="20000" scope="prototype"/>
<!--實施方法注入-->
<bean id="magicBoss" class="com.smart.simple.MagicBoss">
<lookup-method name="getCar" bean="car"/>
</bean>
4.2 方法替換
使用某個Bean的方法替換另外一個Bean的方法
public class Boss1{
public Car getCar{
Car car=new Car();
car.setBrand("寶馬z4");
return car;
}
}
public class Boss2 implements MethodPlacer{
public Object reimplement(Object arg0,Method arg1,Object [] arg2) throws Throwable{
Car car=new Car();
car.setBrand("美人豹");
}
}
<bean id="boss1" class="com.smart.simple.Boss1">
<replace-method name="getCar" replacer="boss2"/>
</bean>
<bean id="boss2" class="com.smart.simple.Boss2"/>
五、Bean之間的關係
5.1 繼承
<bean id="abstractCar" class="com.smart.simple.Car" p:brand="紅旗" p:price="2000" p:color="黑色" abstract="true"/>
<!--abstract=true表示這個<bean>例項化一個對應的bean-->
<bean id="car1" p:color="白色" parent="abstract"/>
5.2 依賴
一般情況下,可以使用<ref>
建立對其他bean的依賴關係。但在某些情況下,要保證依賴項在被依賴項提前初始化,可以使用<depends-on>
顯式指定bean前置依賴的bean。如果前置依賴於多個bean,可以通過逗號空格或分號等方式建立bean的明晨。
<bean id="manager" class="com.smart.simple.Manager" depend-on="sysInt"/>
<bean id="sysInt" class="com.smart.simple.SysInt"/>
5.3 引用
<bean id="car" class="com.smart.simple.Car"/>
<bean id="boss" class="com.smart.simple.Boss">
<property name="carId">
<idref bean="car"/><!--使用idref可以檢查引用關係的正確性,提前發現錯誤-->
</property>
</bean>
六、 整合多個配置檔案
spring允許通過<import>
將多個配置檔案引入到一個檔案中。
現假設有bean1.xml這個配置檔案中有id=”car1”的bean,在bean2.xml中。
<import resource="classpath:com/smart/simple/bean1.xml"/>
<bean id="boss1" class="com.smart.simple.Boss" p:car-ref="car1"/>
七、 Bean作用域
7.1 singleton作用域
在預設情況下,Spring的ApplicationContext容器在啟動時,自動例項化所有singleton的Bean並快取到容器中。如果使用者徐希望在容器啟動時提前例項化singleton的bean,可以通過lazy-init屬性進行控制。
<bean id="boss1" class="com.smart.simple.Boss" p:car-ref="car" lazy-init="true"/>
7.2 prototype作用域
採用scope=”prototype”指定非單例作用域的bean。預設情況下,spring容器在啟動時不例項化prototype的Bean。此外,spring容器將prototype的Bean交給呼叫者後,就不再管理它的生命週期。
7.3 與web應用環境相關的bean作用域
需要進行的額外配置(在高版本的web容器中)
<listener>
<listener-class>
org.springframework.web.context.request.RequestContextListener
</listener-class>
</listener>
<!--
ContextLoaderListener和RequestContextListener的區別
ContenxtLoaderListener:實現了ServletContextListener介面,只負責監聽Web容器啟動和關閉的事件。
RequestContextListener:實現了ServletRequestListener介面,監聽Http請求事件,Web伺服器接受的每一次請求都會通知該監聽器。
-->
7.3.1 request作用域
request作用域的bean對應一個Http請求的宣告週期。每次http請求時會建立一個bean,請求處理完畢後,就銷燬這個bean。
<bean id="car" class="com.smart.simple.Car" scope="request"/>
7.3.2 session作用域
session作用域橫跨整個http session,session中的所用http請求都共享同一個bean,當http session結束後,例項才被銷燬。
<bean id="car" class="com.smart.simple.Car" scope="session"/>
7.3.3 globalSession作用域
globalSession作用域類似於session作用域,不過盡在Portlet的web應用中使用。
<bean id="car" class="com.smart.simple.Car" scope="globalSession"/>
7.4 作用域依賴問題(略)
八、 FactoryBean
當例項化Bean的過程比較複雜,需要在<bean>
中提供大量的配置資訊,可以考慮採用編碼方式例項化bean。
import org.springframework.beans.factory.FactoryBean;
import entity.Car;
public class CarFactoryBean implements FactoryBean<Car>{
private String carInfo;
public String getCarInfo() {
return carInfo;
}
public void setCarInfo(String carInfo) {
this.carInfo = carInfo;
}
@Override
//採用逗號分割的屬性設定資訊
public Car getObject() throws Exception {
Car car=new Car();
String[] infos=carInfo.split(",");
car.setBrand(infos[0]);
car.setColor(infos[1]);
car.setMaxSpeed(Integer.parseInt(infos[2]));
return car;
}
@Override
public Class<Car> getObjectType() {
return Car.class;
}
@Override
public boolean isSingleton() {
return false;
}
}
<bean id="car10" class="chapter5.CarFactoryBean" p:carInfo="紅旗CA72,紅色,12000"/>