上一章主要讲了Spring中Bean的基本装配,包括JavaConfig和XML,以及二者混合使用的方法,本章继续讲解Bean的一些高级装配方法。
在不同的环境中,我们创建某个bean的方式可能不同,比如,在开发阶段,我们可能使用集成的数据库作为DataSource,而在生产阶段则使用不同的DataSource。Spring为这种环境迁移带来的问题提供了很好的解决方案。Spring并不是在构建的时候决定该创建哪个bean或不创建哪个bean,而是在运行的时候再来确定。因此,同一个部署单元(可能是一个WAR包)能够适用于所有的环境。
Spring使用注解@Profile来确定Bean是否被创建。只有当@Profile中的profile激活时bean才会创建,如果没有激活,带有@Bean注解的方法都会被忽略掉
在Spring3.1中,只能在类级别上使用@Profile, 从Spring3.2开始,可以在方法级别上使用@Profile注解。
package com.example.config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Profile;
import org.springframework.jdbc.datasource.embedded.EmbeddedDatabaseBuilder;
import org.springframework.jdbc.datasource.embedded.EmbeddedDatabaseType;
import org.springframework.jndi.JndiObjectFactoryBean;
import javax.sql.DataSource;
@Configuration
public class DataSourceConfig {
@Bean(destroyMethod = "shutdown")
@Profile("dev") //只有dev profile激活时这个bean才会创建,用于开发环境
public DataSource embeddedDataSource() {
return new EmbeddedDatabaseBuilder().setType(EmbeddedDatabaseType.H2)
.addScript("classpath:schema.sql")
.addScript("classpath:test-data.sql")
.build();
}
@Bean
@Profile("prod") //prod profile激活时,bean才会创建,用于生产环境
public DataSource jndiDataSource() {
JndiObjectFactoryBean jndiObjectFactoryBean = new JndiObjectFactoryBean();
jndiObjectFactoryBean.setJndiName("jdbc/myDs");
jndiObjectFactoryBean.setResourceRef(true);
jndiObjectFactoryBean.setProxyInterface(javax.sql.DataSource.class);
return (DataSource)jndiObjectFactoryBean.getObject();
}
}
使用XML配置的方式本文不再讲解。
Spring依靠spring.profiles.active和spring.profiles.default来确定哪个profile处于激活状态。如果设置了spring.profiles.active属性的话,它的值就会用来确定哪个profile是激活的;没有设置的话,spring就会查找spring.profiles.default。
有多种方式来设置者两个属性:
@Conditional注解可以用来给bean的创建设置条件,必须满足条件bean才会创建。
Spring提供了多种可选方案来解决装配bean时的歧义性,可以将可选bean中的某一个设为首选(primary)的bean,或者使用限定符(qualifier)来帮助Spring将可选的bean的范围缩小到只有一个bean。
标示首选的bean:
@Component
@Primary
public class BlankDisc implements CompactDisc{ ... }
或者在JavaConfig中显示地声明BlankDisc:
@Bean
@primary
public BlankDisc blankDisc() {
return new BlankDisc();
}
使用XML配置
id="blankDisc" class="com.example.chr1s.BlankDisc" primary="true" />
当标识了多个首选bean时,还可以使用限定符来限定bean,限定符的功能更加强大,也更通用。
@Qualifier注解可以与@Autowired和@Inject注解系统使用。@Qualifier注解还有很多高级的应用,但是这些应用在实际开发中使用到的比较少,本文不再详述。
@Autowired
@Qualifier("blankDisc")
public void setDisc(CompactDisc compactDisc){
disc = compactDisc;
}
为@Qualifier注解设置的参数就是想要注入的bean的id。
默认情况下,Spring应用上下文中所有的bean都是作为以单例的形式创建的。也就是说,不管给定的一个bean被注入到其他bean多少次,每次所注入的都是同一个实例。
单例形式不是线程安全的,在很多场合下不适用。Spring还定义了多种作用域,可以基于这些作用域创建bean,包括:
要选择不同的作用域,应该使用@Scope注解。
@Component
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public class NotePad { ... }
将bean的作用域声明为prototype。
也可以在配置文件中声明作用域
@Bean
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public NotePad notePad() {
return new NotePad();
}
@Scope有一个proxyMode属性,解决了将会话作用域的bean注入到单例bean中所遇到的问题。后面会进行更加详细的讲解。
Spring提供两种在运行时求值的方法:
可以通过声明属性源并通过Spring的Environment来检索属性,可以使用@PropertySource和Environment来实现。