《Spring实战 第三版》二
第二章 装配Bean
在Spring中,对象无需自己负责查找或创建与其相关联的其他对象
相反,容器负责把需要相互协作的对象引用赋予各个对象
创建应用对象之间协作关系的行为通常称为装配,这也是依赖注入的本质
声明Bean
Spring是一个基于容器的框架
但是如果没有对Spring进行配置,那它就是一个空容器,不起任何作用
所以我们需要配置Spring来告诉它需要加载哪些Bean和如任何装配这些Bean
这样才能确保它们能够彼此协作
从Spring3.0开始,Spring容器提供了两种配置Bean的方式
一个是:一个或多个XML文件;另一个是:基于Java注解的配置方式
我们首先来关注传统的XML文件配置方式
1.创建Spring配置
在XML文件中声明Bean时,Spring配置文件的根元素来源于Spring beans命名空间所定义的<beans>元素
典型的Spring XMl配置文件:
<?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:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!--Bean declaration go here--> </beans>
在根元素<beans>中,可以放置所有的Spring配置信息,包括<bean>元素的声明
2.声明一个简单的Bean
假设我们有一位杂技师,可以同时抛多个豆袋子
public class Juggler implements Performer{ private int beanBags=3; public Juggler(){} public Juggler(int beanBags){ this.beanBags = beanBags; } public void perform(){ System.out.println("JUGGLING "+beanBags+" BEANBAGS!"); } }
我们需要在XML配置文件中声明:<bean id="duke" class="Jugger"></bean>
<bean>元素是Spring中最基本的配置单元,通过该元素Spring将创建一个对象
这里创建了一个由Spring容器管理的名字为duke的Bean,id属性定义了Bean的名字
也作为该Bean在Spring容器中的引用,class属性则告诉我们duke是一个Jugger
当Spring容器加载该Bean时,Spring将使用默认的构造器来实例化duke
即duke将可以同时抛三袋豆袋子
3.通过构造器注入
那么,我们若是想要duke可以抛更多的豆袋子呢?比如抛15个豆袋子该怎么实现呢?
<bean id="duke" class="Jugger"> <constructor-arg value="15"> </bean>
在构造Bena时,我们可以使用<constructor-arg>元素来告诉Spring额外的信息
这时,Jugger将调用另外一个构造方法,value将值传递给了BeanBags
4.通过构造器注入对象引用
如果,我们还想让duke一边抛豆袋子一边朗诵莎士比亚的诗歌呢?
先对Jugger类进行修改
public class PoeticJuggler extends Juggler { //这里Poem是一个接口,该接口声明了recite方法 private Poem poem; public PoeticJuggler(int beanBags,Poem poem){ super(beanBags); this.poem = poem; } public void perform(){ super.perform(); System.out.println("While reciting..."); poem.recite(); } }
同时,我们假设有一个类Sonnet29它实现了Poem接口,并定义了recite方法
用recite方法来朗诵莎士比亚的诗歌,并在XML文件中对它进行了声明<bean id="sonnet29" class="Sonnet29"></bean>
回到问题本身来,那么怎么才能让duke变朗诵诗歌边抛豆袋子呢?
需要在XML文件中做如此声明即可
<bean id="duke" class="PoeticJuggler"> <constructor-arg value="15"/> <constructor-arg ref="sonnet29"/> </bean>
这里,Poem不是简单类型,不能使用value
需要使用ref属性来实现从构造器注入对象引用
从前面的例子可以看出,通过构造器注入来创建Bean是挺不错的
5.通过工厂方法创建Bean
但是如果你想声明的Bean没有一个公开的构造方法,那怎么办呢?
那就需要了解如何装配通过工厂方法创建的Bean
考虑下这种场景,在Spring中将一个单例类配置为Bean
一般来说,单例类的实例只能通过静态工厂方法来创建
public class Stage { private Stage(){} private static class StageSingletonHolder{ static Stage instance = new Stage(); } public static Stage getInstance(){ return StageSingletonHolder.instance; } }
这里Stage没有一个公开的构造方法
取而代之的是,静态方法getInstance()每次调用时都返回相同的实例
在Spring中如何把没有公开构造方法的Stage配置为一个Bean呢?<bean id="theStage" class="Stage" factory-method="getInstance"/>
幸好,bean元素有一个factory-method属性,允许我们调用一个指定的静态方法
从而代替构造方法来创建一个类的实例
6.Bean的作用域
所有的Spring Bean默认都是单例,即当容器分配一个Bean时,总是返回Bean的同一个实例
那么如何让Spring在每次请求时都为Bean产生一个新的实例?<bean id="duke" class="Jugger" scope="prototype">
除了prototype,Spring还提供了其他几个作用域选项
7.初始化和销毁Bean
当实例化一个Bean时,可能需要执行一些初始化操作
当不再需要Bean时,可能要做一些清除工作
为Bean定义初始化和销毁操作,只需要使用init-method和destroy-method参数来配置bean元素
init-method属性指定了在初始化Bean时要调用的方法
destroy-method属性指定了Bean从容器移除之前要调用的方法
假设有一个礼堂类Auditorium,它需要在活动开始前开灯和活动结束后关灯
为了实现该需求我们可以使用init-method和destroy-method属性来声明auditorium Bean:
<bean id="auditorium" class="Auditorium" init-method="turnOnLights" destroy-method="turnOffLights" />
auditorium Bean实例化后会立即调用turnOnLights方法
该Bean从容器中移除和销毁前,会调用turnOffLights方法
注入Bean属性
通常JavaBean的属性是私有的,同时拥有一组存储器方法
以setXXX和getXXX形式存在
Spring可以借助属性的set方法来配置属性的值,已实现settet方式的注入
1.setter方式的注入
假设我们有一位才华横溢的音乐家
public class Instrumentalist implements Performer { public Instrumentalist(){} @Override public void perform() { System.out.println("Playing"+song+":"); instrument.play(); } private String song; //注入歌曲 public void setSong(String song) { this.song = song; } //Instrument是一个接口 private Instrument instrument; //注入乐器 public void setInstrument(Instrument instrument){ this.instrument = instrument; } }
我们在XML文件中对其进行声明,并调用属性的setter方法来注入song
<bean id="kenny" class="Instrumentalist"> <property name="song" value="Jingle Bell"/> <!--Saxophone类实现了Instrucment接口,且其Bean的ID为saxophone--> property name="instrument" ref="saxophone"/> </bean>
一旦Instrumentalist被实例化,Spring就会调用<property>元素所指定属性的setter方法为该属性注入值
我们还可以注入内部Bean,即定义在其他Bean内部的Bean
<bean id="kenny" class="Instrumentalist"> <property name="song" value="Jingle Bell"/> <property name="instrument"> <bean class="Saxophone"/> </property> </bean>
内部Bean是没有ID属性的,也就是说不能被复用
内部Bean仅适用于一次注入,而且也不能被其它Bean所引用
2.使用Sprin的命名空间p装配属性
Spring的空间命名p还提供了另外一种Bean属性装配的方式
该方式不需要配置如此多的尖括号
只需要在Spring的XML文件中增加这一声明即可xmlns:p="http://www.springframework.org/schema/p"
通过该声明,可以使用p:作为<bean>元素所有属性的前缀来装配Bean的属性
<bean id="kenny" class="Instrumentalist" p:song="Jingle Bell" p:instrument-ref="saxophone" />
命令空间p的最主要优点是更简洁
3.装配集合
如果属性的类型是集合,Spring该如何配置呢?
假设有一个非常厉害的人——hank,可以在同一时刻弹奏多种乐器
public class OneManBand implements Performer { public OneManBand(){} @Override public void perform() { for(Instrument instrument:instrucments){ instrument.play(); } } private Collection<Instrument> instrucments; //注入Instrument集合 public void setInstrucments(Collection<Instrument> instrucments) { this.instrucments = instrucments; } }
为了让hank可以同时表演多个乐器,我们需要使用<list>来配置元素
<list>可以用来包含一个或者多个值
<bean id="hank" class="OneManBand"> <property name="instrucments"> <list> <ref bean="saxophone"/> <ref bean="piano"/> </list> </property> </bean>
同样地,也可以使用<set>元素来装配集合或者数组类型的属性
无论是还是都可以用来装配类型为java.util.Collection的任意实现或者数组的属性
那么Map集合又该如何装配呢?
将OneManBand的私有属性instrucments进行修改
private Map<String,Instrument> instrucments; public void setInstrucments(Map<String, Instrument> instrucments) { this.instrucments = instrucments; }
那么对应的XML配置文件该修改为:
<bean id="hank" class="OneManBand"> <property name="instrucments"> <map> <entry key="Saxo" value-ref="saxophone"/> <entry key="Piano" value-ref="piano"/> </map> </property> </bean>
如果所配置的Map的每一个entry的键和值都是String类型时,可以使用java.util.Properties
同样地,将OneManBand的私有属性instrucments进行修改
private Properties instrucments; public void setInstrucments(Properties instrucments) { this.instrucments = instrucments; }
对应的XML文件应修改为:
<bean id="hank" class="OneManBand"> <property name="instrucments"> <props> <prop key="Saxo">saxophone</prop> <prop key="Piano" >piano/</prop> </props> </property> </bean>
5.装配空值
Spring还可以为装配一个空值
为属性设置null值,只需使用<null/>元素<property name="..." ><null/></property>
使用表达式装配
到目前为止,我们所有的关于Bean的装配都是在Spring的XML文件中静态定义的
但是,如果我们为属性装配的值只有运行期才能知道,那有如何实现呢?
Spring 3 引入了Spring表达式语言(Spring Expression Language,SpEL)
SpEL通过运行期间执行的表达式将值装配到Bean的属性或者构造器参数中
#{}标记会提示Spring这个标记内容是SpEL表达式<property name="instrucment" value="#{saxophone}">
在这里我们使用SpEL把一个ID为“saxophone”的Bean装配到了instrument属性中
SpEL能做的不仅仅是引用Bean,还可以使用Bean的引用来获取属性
也可以通过Bean的引用来调用方法
在SpEl中,还可以使用T()运算符来调用类作用域的方法和常数
如获取Java中的Math类的PI#{(java.lang.Math).PI}
除此之外,还可以使用SpEL来进行运算