Profile


总有那么一些时候,我们希望Spring容器能够根据我们提供的条件决定哪些Bean需要创建,哪些Bean不需要创建。提供的条件不同,Spring容器创建的Bean也不同。创建的Bean不同,软件实现的功能自然也有所差别。也就是说,我们希望在某些应用场景下无需修改代码或重新构建项目,只需简单修改一下条件就能达到改变软件功能的目的。

于是,Profile出现了。能让我们向Spring容器提供一些配置信息,告诉Spring容器两件事情:
1.告诉Spring容器我们想要创建的Bean属于哪个Profile
2.告诉Spring容器只需创建属于某些Profile的Bean,无需创建属于其它Profile的Bean

如此一来,Spring容器就能按照指定的Profile创建指定的Bean。指定的Profile不同,创建的Bean也不同。创建的Bean不同,软件实现的功能自然也就不同。非常明显,Profile就是专门用来条件化Bean的创建的。

问题在于,我们应该怎样告诉Spring容器Bean是属于哪个Profile的呢?

这就涉及@Profile注解了。@Profile注解有个value属性,能够指定Bean所属的Profile。假如com.dream包现有这样一些类:

 1 public interface Music {
 2 }
 3 
 4 public class ClassicMusic implements Music {
 5 }
 6 
 7 public class CountryMusic implements Music {
 8 }
 9 
10 public class Player {
11     private Music playingMusic = null;
12 
13     public Player(Music playingMusic) {
14         this.playingMusic = playingMusic;
15     }
16 }

则可使用@Profile注解这样配置Bean:

 1 @Configuration
 2 public class AppConfig {
 3     @Bean(name = "classisMusic")
 4     @Profile(value = "classicProfile")
 5     public Music produceClassicMusic() {
 6         return new ClassicMusic();
 7     }
 8 
 9     @Bean(name = "countryMusic")
10     @Profile(value = "countryProfile")
11     public Music produceCountryMusic() {
12         return new CountryMusic();
13     }
14 
15     @Bean(name = "player")
16     public Player producePlayer(Music playingMusic) {
17         return new Player(playingMusic);
18     }
19 }

该配置类定义了三个方法,配置了三个Bean:
1.方法produceClassicMusic带有 @Profile(value = "classicProfile") 注解。如果激活的是classicProfile,Spring容器就会调用这个方法创建ClassicMusic类型的Bean;否则不创建。
2.方法produceCountryMusic带有 @Profile(value = "countryProfile") 注解。如果激活的是countryProfile,Spring容器就会调用这个方法创建CountryMusic类型的Bean;否则不创建。
3.方法producePlayer没带@Profile注解,不属于任何Profile。不管激活的是哪些Profile,Spring容器都会调用这个方法创建Player类型的Bean

于是我们知道了,Bean所属的Profile是由@Profile注解标注的。接下来我们需要做的,就是告诉Spring容器哪些Profile是激活的。而这,还得仰赖Spring提供的Environment接口。如下所示:

1 try (var context = new AnnotationConfigApplicationContext()) {
2     context.getEnvironment().setActiveProfiles("classicProfile");
3     context.register(AppConfig.class);
4     context.refresh();
5 }

这段代码做了这些事情:
1.创建AnnotationConfigApplicationContext类型的Spring容器。
2.获取Spring容器里的Environment接口之后调用setActiveProfiles方法激活Profile
3.把Java配置类注册给Spring容器。
4.刷新Spring容器,完成Bean的创建。

毫无疑问,问题的关键在于第二步。Environment接口提供了setActiveProfiles方法,用于激活指定的Profile。它的签名如下:

1 public void setActiveProfiles(String... profiles)

它的参数是 String... 类型的,能够同时指定多个激活的Profile。而我们的这段代码只激活了classicProfile,告诉Spring容器只创建ClassicMusic类型的Bean,不创建CountryMusic类型的Bean。当然,Player类型的Bean总会被创建,因为它不属于任何Profile

值得关注的是,除了激活Profile,我们也能指定默认的Profile。这样,如果指定了激活的Profile,Spring容器就会使用激活的Profile创建Bean;如果没有指定激活的Profile,Spring容器就会使用默认的Profile创建Bean。默认的Profile能由Environment接口的setDefaultProfiles方法指定。它的签名如下:

1 public void setDefaultProfiles(String... profiles)

它的参数是 String... 类型的,同样支持同时指定多个默认的Profile,这和setActiveProfiles方法是一样的,不再赘叙。

于是我们知道了,默认或激活的Profile都可调用Environment接口指定。而在Web开发中,我们并不需要直接调用Environment接口,只要使用spring.profiles.default属性指定默认的Profile,使用spring.profiles.active属性指定激活的Profile就行。默认或激活的Profile可以同时指定多个,之间用逗号隔开即可。

这样,Spring容器就能用spring.profiles.default属性的值调用Environment接口指定默认的Profile;使用pring.profiles.active属性的值调用Environment接口指定激活的Profile。以下是spring.profiles.default属性和spring.profiles.active属性可以指定的地方:
1.Servlet初始化参数
2.Servlet上下文初始化参数
3.JNDI环境变量
4.JVM系统属性(也就是JVM的命令行参数)
5.操作系统环境变量

比如,我们可在部署描述文件web.xml里指定Servlet初始化参数和Servlet上下文初始化参数,这样指定默认的Profile:

 1 <context-param>
 2   <param-name>contextConfigLocationparam-name>
 3   <param-value>/WEB-INF/config/root-config.xmlparam-value>
 4 context-param>
 5 <context-param>
 6   <param-name>spring.profiles.defaultparam-name>
 7   <param-value>classicProfileparam-value>
 8 context-param>
 9 <listener>
10   <listener-class>
11     org.springframework.web.context.ContextLoaderListener
12   listener-class>
13 listener>
14 
15 <servlet>
16   <servlet-name>dispatcherServletservlet-name>
17   <servlet-class>org.springframework.web.servlet.DispatcherServletservlet-class>
18   <init-param>
19     <param-name>contextConfigLocationparam-name>
20     <param-value>/WEB-INF/config/servlet-config.xmlparam-value>
21   init-param>
22   <init-param>
23     <param-name>spring.profiles.defaultparam-name>
24     <param-value>classicProfileparam-value>
25   init-param>
26   <load-on-startup>1load-on-startup>
27 servlet>

在Servlet初始化参数里,我们把classicProfile指给spring.profiles.default属性,以使根容器知道默认的Profile是classicProfile,进而只创建属于classicProfile的Bean;在Servlet上下文初始化参数里,我们把classicProfile指给spring.profiles.default属性,以使Servlet容器知道默认的Profile是classicProfile,进而只创建属于classicProfile的Bean

到了运行Web应用程序的时候,如果我们想要创建属于countryProfile的Bean,只需通过JVM命令行参数之类的,把countryProfile指给spring.profiles.active属性就行。

还有,@Profile注解除了可以像上面的示例那样加到配置方法之外,也可以加到配置类上。如下所示:

 1 @Configuration
 2 @Profile(value = "classicProfile")
 3 public class AppConfigClassic {
 4     @Bean(name = "classisMusic")
 5     public Music produceClassicMusic() {
 6         return new ClassicMusic();
 7     }
 8 }
 9 
10 @Configuration
11 @Profile(value = "countryProfile")
12 public class AppConfigCountry {
13     @Bean(name = "countryMusic")
14     public Music produceCountryMusic() {
15         return new CountryMusic();
16     }
17 }
18 
19 @Configuration
20 public class AppConfig {
21     @Bean(name = "player")
22     public Player producePlayer(Music playingMusic) {
23         return new Player(playingMusic);
24     }
25 }

这段代码定义了三个配置类:
1.配置类AppConfigClassic带有 @Profile(value = "classicProfile") 注解。只有激活了classicProfile,Spring容器才会创建该配置类里的Bean;否则不创建。
2.配置类AppConfigCountry带有 @Profile(value = "countryProfile") 注解。只有激活了countryProfile,Spring容器才会创建该配置类里的Bean;否则不创建。
3.配置类AppConfig没带@Profile注解。不管激活了哪些Profile,Spring容器都会创建该配置类里的Bean

当然,XML配置文件也支持Profile。如果采用XML配置文件的话,可用元素这样配置:

 1 <beans /* 省略命名空间和XSD模式文件声明 */>
 2     <beans profile="classicProfile">
 3         <bean id="classicMusic" class="com.dream.ClassicMusic">
 4             <property name="musicName" value="Classic Music" />
 5         bean>
 6     beans>
 7     <beans profile="countryProfile">
 8         <bean id="countryMusic" class="com.dream.CountryMusic">
 9             <property name="musicName" value="Country Music" />
10         bean>
11     beans>
12 
13     <bean id="player" class="com.dream.Player" autowire="byType" />
14 beans>

也可使用多个XML配置文件这样配置:

 1 <beans profile="classicProfile" /* 省略命名空间和XSD模式文件声明 */>
 2     <bean id="classicMusic" class="com.dream.ClassicMusic">
 3         <property name="musicName" value="Classic Music" />
 4     bean>
 5 beans>
 6 
 7 <beans profile="countryProfile" /* 省略命名空间和XSD模式文件声明 */>
 8     <bean id="countryMusic" class="com.dream.CountryMusic">
 9         <property name="musicName" value="Country Music" />
10     bean>
11 beans>
12 
13 <beans /* 省略命名空间和XSD模式文件声明 */>
14     <bean id="player" class="com.dream.Player" autowire="byType" />
15 beans>

至此,关于Profile的介绍也就告一段落了。下章,我们将会开始介绍Condition以及藏在Profile背后的秘密。欢迎大家继续阅读,谢谢大家!

    下载代码