Spring中如何装配Bean
在Spring中,对象无需自己查找或者创建与它相关联的其他对象,由容器负责将需要相互协作的对象引用传递给各个对象。容器所做的这个工作就是依赖注入的本质,这是Spring提供的最基本的功能。
Spring是一个基于容器的框架,如果没有配置,它就是一个没有任何作用的空容器。从Spring3.0开始,Spring提供了两种配置Bean的方式:使用XML文件配置和使用注解配置。
表1 Spring核心框架自带的xml命名空间
命名空间
用途
aop
为声明切面以及将@AspectJ注解的类代理为Spring切面提供了配置元素
beans
声明bean和装配bean
context
配置Spring应用上下文,自动检测、自动装配bean,注入非Spring直接管理的对象
jee
提供了与java ee API的集成,例如JNDI和EJB
jms
为声明消息驱动的pojo提供了配置元素
lang
支持由Groovy、JRuby和BeanShell等脚本实现的bean
mvc
启用Spring MVC
oxm
支持Spring的对象到xml映射配置
tx
提供声明式事务配置
util
提供工具类元素
实例:
整个工程在github上:https://github.com/simon166/demo/tree/master/springAction
一个Performer接口:
package springAction;
public interface Performer {
void perform();
}
一个Juggler类实现了Performer接口:
package springAction;
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");
}
}
一个 PoeticJuggler类继承了Juggler类:
package springAction;
public class PoeticJuggler extends Juggler {
private Poem poem;
public PoeticJuggler(Poem poem){
super();
this.poem = poem;
}
public PoeticJuggler(int beanBags, Poem poem){
super(beanBags);
this.poem = poem;
}
public void perform(){
super.perform();
System.out.println("PoeticJugger reciting...");
poem.recite();
}
}
一个Poem接口:
package springAction;
public interface Poem {
void recite();
}
一个单例类Stage:
package springAction;
public class Stage {
private Stage(){
}
private static class StageSingletonHolder{
static Stage instance = new Stage();
}
public static Stage getInstance(){
return StageSingletonHolder.instance;
}
}
看看xml文件怎么写(springAction.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"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.1.xsd">
<!-- 配置一个bean -->
<bean id="duke" class="springAction.Juggler" >
<!--给构造函数传递参数,没有的话则调用默认构造方法 -->
<constructor-arg value="15"/>
</bean>
<bean id="sonnet29" class="springAction.Sonnet29"></bean>
<bean id="poeticDuke" class="springAction.PoeticJuggler">
<constructor-arg value="22"></constructor-arg>
<!-- 基本数据类型参数用value=字符串(Spring根据构造参数类型自动解析字符串),
引用类型的参数用ref= bean id -->
<constructor-arg ref="sonnet29"></constructor-arg>
</bean>
<!-- factory-method通过工厂方法将单例类配置为bean,因为Stage没有构造函数 -->
<bean id="theStage" class="springAction.Stage" factory-method="getInstance">
</bean>
</beans>
加个main函数运行一下,看看效果:
package springAction;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class springActionMain {
public static void main(String[] args) {
// TODO Auto-generated method stub
ApplicationContext ctx = new ClassPathXmlApplicationContext(
"springAction/springAction.xml");
try{
Performer performer = (Performer)ctx.getBean("duke");
performer.perform();
performer = (Performer)ctx.getBean("poeticDuke");
performer.perform();
}catch(Exception e){
e.printStackTrace();
}finally{
((ClassPathXmlApplicationContext)ctx).close();
}
}
}
效果:
Juggling 15 beanbags
Juggling 22 beanbags
PoeticJugger reciting…
balblabla…
以上是一个xml文件配置bean的简单例子,下面看看bean的作用域。
bean的作用域
所有的Spring Bean默认是单例,当容器分配一个Bean时,它总是返回同一个实例。但,spring是灵活的,不需要单例模式时,可以如下配置:
<bean id="ticket" class="省略号" scope="prototype"/>
scope表示作用域,prototype表示每次返回不同的实例。bean的作用域还有:
作用域
简介
singleton
单例模式(默认)
prototype
每次返回不同的实例
request
该作用域仅用于基于web的spring上下文(spring mvc),在一个HTTP 请求中每个bean定义对应一个实例
session
该作用域仅用于基于web的spring上下文(spring mvc),在一个HTTP Session中每个bean定义对应一个实例
global-session
该作用域仅在Portlet上下文中才有效,在一个全局HTTP Session中每个bean定义对应一个实例
初始化Bean和销毁Bean
当Spring容器实例化一个Bean时可能需要做一些初始化的工作,当Spring销毁一个Bean时,可能需要做一些清理工作。Spring支持Bean提供自定义的方法来做初始化工作和销毁工作。
<bean id="beanName"
class="classPath"
init-method="init_method_in_bean" bean中的初始化方法
destroy-method="destory_method_in_bean"/> bean中的销毁方法
如上配置,Spring会自动的在实例化该bean时调用init-method指定的初始化方法,在销毁该bean时调用destory-method指定的销毁方法。
为Bean注入属性
一般地,java Bean会有一些私有属性,并有一些set和get方法。Spring可以利用set方法为bean的私有属性注入值。
例子:
package springAction;
public class Instrumentalist implements Performer{
private String song;
private Instrument instrument;
private int age;
public Instrumentalist(){
}
public void perform(){
System.out.println("age = " + age);
System.out.println("Playing "+song);
instrument.play();
}
public String getSong() {
return song;
}
public void setSong(String song) {
this.song = song;
}
public Instrument getInstrument() {
return instrument;
}
public void setInstrument(Instrument instrument) {
this.instrument = instrument;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
xml文件配置:
<bean id="saxophone" class="springAction.Saxophone"/>
<bean id="kenny" class="springAction.Instrumentalist">
<!-- 配置属性song -->
<property name="song" value="ABC"></property>
<!-- 配置属性age,虽然此处是字符串“14”,但是Spring会识别age的类型,然后把字符串“14”转变后赋值给age -->
<property name="age" value="14"></property>
<!-- 配置instrument对象,用ref的方式赋引用,和构造函数一致 -->
<property name="instrument" ref="saxophone"></property>
</bean>