前言
bean
是Spring
最基础最核心的部分,Spring
简化代码主要是依赖于bean
,下面学习Spring
中如何装配bean
。
装配bean
Spring
在装配bean
时非常灵活,其提供了三种方式装配bean
。
- 在
XML
中进行显式配置。 - 在
Java
中进行显式配置。 - 隐式的
bean
发现机制和自动装配。
自动化装配bean
自动化装配技术最为便利,
Spring
从两个角度实现自动化装配。
- 组件扫描:
Spring
会自动发现应用上下文中所创建的bean
。 - 自动装配:
Spring
自动满足bean
之间的依赖。
自动装配示例
- pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.hust.grid.leesf.spring</groupId>
<artifactId>spring-learning</artifactId>
<version>1.0-SNAPSHOT</version>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<spring.version>3.1.2.RELEASE</spring.version>
<cglib.version>3.1</cglib.version>
<junit.version>4.11</junit.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>${junit.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib-nodep</artifactId>
<version>${cglib.version}</version>
</dependency>
</dependencies>
</project>
- CompactDisc
package ch2;
interface CompactDisc {
void play();
}
其只定义了一个
play
接口,由子类实现。
- SgtPeppers
package ch2;
import org.springframework.stereotype.Component;
@Component
public class SgtPeppers implements CompactDisc {
private String title = "Sgt. Pepper's Lonely Hearts Club Band";
private String artist = "The Beatles";
public void play() {
System.out.println("Playing " + title + " by " + artist);
}
}
SgtPeppers
继承了CompactDisc
接口,使用Component
注释为一个Bean
。
- CDPlayerConfig
package ch2;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
@Configuration
@ComponentScan("ch2")
public class CDPlayerConfig {
}
配置类,
Spring
会自动加载上下文并扫描ch2
包下的所有bean
。
- CDPlayerTest
package ch2;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import static org.junit.Assert.assertNotNull;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = CDPlayerConfig.class)
public class CDPlayerTest {
@Autowired
private CompactDisc cd;
@Test
public void cdShouldNotNull() {
assertNotNull(cd);
}
}
该类用于测试是否成功装配
CompactDisc
的bean
,测试成功。
设置Bean名称
上述示例中
bean
的名称默认为sgtPeppers
,即将类名的首字母小写,当然可通过@Component("sp")
设置其名称为sp
;或者使用@Named("sp")
进行设置。
设置组建扫描基础包
上述示例中指定扫描
ch2
包,这是通过@ComponentScan("ch")
指定的,当然可以通过@ComponentScan(basePackages="ch2")
进行设置。若设置多个包,则可采用@ComponentScan(basePackages={"ch2","video"})
方式进行设置。除了使用字符串格式表明包名,也可使用类名格式,如@ComponentScan(basePackageClasses = SgtPeppers.class)
指定扫描类。
设置自动装配
示例中使用
@Autowired
实现自动装配,Spring
应用上下文中寻找某个匹配的bean
,由于示例中CompactDisc
只有一个声明为bean
的子类,若有多个声明为bean
的子类,则会报错,若没有子类,也会报错。@Autowired
注解不仅可以用在属性上,也可以用在构造函数上,还可以用在Setter方法上。若使用@Autowired(required=false)
时,那么没有符合的bean
时不会报错,而是处于未装配的状态,要防止空指针情况,其与@Inject
注解功能类似。
- 构造函数使用
@Autowired
注解
@Component
public class CDPlayer implements MediaPlayer {
private CompactDisc cd;
@Autowired
public CDPlayer(CompactDisc cd) {
this.cd = cd;
}
public void play() {
}
}
Setter
方法使用@Autowired
注解
@Autowired
public void setCompactDisc(CompactDisc cd) {
this.cd = cd;
}
在Java中显式配置
在配置类中显式配置
bean
,将CDPlayerConfig
中的@ComponentScan("ch2")
移除,此时运行测试用例会报错,下面通过显式配置方法来配置bean
。修改CDPlayerConfig
代码如下。
package ch2;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class CDPlayerConfig {
@Bean
public CompactDisc sgtPeppers() {
return new SgtPeppers();
}
}
上述生成的
bean
名称与方法名相同,若想设置名称,可通过@Bean(name=sp)
进行设置。对于如下代码,调用sgtPeppers
会生成同一个sgtPeppers
的bean
,这是由于sgtPeppers
方法标记为Bean
,Spring
会拦截所有对该方法的调用,并且返回一个已经创建的bean
实例。默认情况下,Spring
中的bean
都是单例的。
@Bean
public CDPlayer cdPlayer() {
return new CDPlayer(sgtPeppers());
}
@Bean
public CDPlayer anotherCDPlayer() {
return new CDPlayer(sgtPeppers());
}
还可以使用如下方法来引用bean
@Bean
public CDPlayer cdPlayer(CompactDisc compactDisc) {
return new CDPlayer(compactDisc);
}
这样会自动装配一个CompactDisc
到配置方法中,不用明确使用sgtPeppers
方法来构造CDPlayer
。
通过xml装配bean
除了使用
JavaConfig
来显式装配bean
外,还可以使用xml
文件来装配bean
。若想在xml
中声明一个bean
元素,则需要如下操作。
<?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:context="http://www.springframework.org/schema/context"
xmlns:c="http://www.springframework.org/schema/c"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<bean id="sgtPeppers" class="ch2.SgtPeppers" />
</beans>
上述xml
文件中声明了一个名为sgtPeppers
的bean
,会调用SgtPeppers
的默认构造函数创建bean
。
使用构造器注入初始化bean
使用constructor-arg元素
<bean id="cdPlayer" class="ch2.CDPlayer">
<constructor-arg ref="compactDisc"/>
</bean>
上述代码表示将ID
为compactDisc
的bean
引用传递到CDPlayer
的构造器中。
使用c-命令空间
<bean id="cdPlayer" class="ch2.CDPlayer"
c:cd-ref="compactDisc"/>
</bean>
其中c:
表示命名空间前缀;cd
表示构造器参数名;-ref
表示注入的bean
的引用;compactDisc
表示要注入的bean
的ID
。
将字面量注入到构造器中
<bean id="compactDisc" class="ch2.BlankDisc">
<constructor-arg value="Sgt. Pepper's Lonely Hearts Club Band" />
<constructor-arg value="The Beatles" />
或者使用
<bean id="compactDisc" class="ch2.BlankDisc"
<c:_title="Sgt. Pepper's Lonely Hearts Club Band" />
<c:artist="The Beatles" />
装配集合到构造器中
装配字面量到List集合
<bean id="compactDisc" class="ch2.BlankDisc">
<constructor-arg value="Sgt. Pepper's Lonely Hearts Club Band" />
<constructor-arg value="The Beatles" />
<constructor-arg>
<list>
<value>Sgt.AA</value>
<value>Sgt.BB</value>
<value>Sgt.CC</value>
</list>
</constructor-arg>
</bean>
装配引用List集合
<bean id="compactDisc" class="ch2.BlankDisc">
<constructor-arg value="Sgt. Pepper's Lonely Hearts Club Band" />
<constructor-arg value="The Beatles" />
<constructor-arg>
<list>
<ref bean="sgtPeppers"/>
<ref bean="whiteAlbum"/>
<ref bean="revolver"/>
</list>
</constructor-arg>
</bean>
同理,对于Set
集合只需修改list
为set
即可。
设置属性
使用xml设置属性
<bean id="cdPlayer" class="ch2.CDPlayer">
<property name="compactDisc" ref="compactDisc">
</bean>
使用p-命令空间进行装配
<bean id="cdPlayer" class="ch2.CDPlayer"
p:compactDisc-ref="compactDisc">
</bean>
其组成与c-
类似。
将字面量装配到属性中
<bean id="compactDisc" class="ch2.BlankDisc">
<property name="title" value="Sgt. Pepper's Lonely Hearts Club Band" />
<property name="artist"value="The Beatles" />
<property name="tracks">
<list>
<value>Sgt.AA</value>
<value>Sgt.BB</value>
<value>Sgt.CC</value>
</list>
</property>
</bean>
使用p-装配属性
<bean id="compactDisc" class="ch2.BlankDisc">
<p:title="Sgt. Pepper's Lonely Hearts Club Band" />
<p:artist="value="The Beatles" />
<property name="tracks">
<list>
<value>Sgt.AA</value>
<value>Sgt.BB</value>
<value>Sgt.CC</value>
</list>
</property>
</bean>
使用util-命名空间装配集合
<util:list id="tractList">
<value>Sgt.AA</value>
<value>Sgt.BB</value>
<value>Sgt.CC</value>
</util:list>
此时对应修改如下
<bean id="compactDisc" class="ch2.BlankDisc">
<p:title="Sgt. Pepper's Lonely Hearts Club Band" />
<p:artist="value="The Beatles" />
<p:tracks-ref="trackList" />
</bean>
导入和混合配置
在JavaConfig中引用xml配置
将
BlankDisc
从CDPlayerConfig
中剥离出来,放置在自己的配置文件CDConfig
中。此时需要在CDPlayerConfig
中使用@Import(CDConfig.class)
将两者组合;或者使用更高级别的Config
中使用@Import({CDPlayerConfig.class,CDConfig.class})
组合两者。若将BlankDisc
配置在cd-config.xml
文件中,则可使用@ImportResource("classpath:cd-config.xml")
导入。
在xml配置中引用JavaConfig
可以使用
import
元素引用配置,如
<import resource="cd-config.xml" />
总结
Spring
有三种方式装配bean
,使用自动化装配技术使得代码更简洁;并且有多种方式注入属性。