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包现有这样一些类:

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

该配置类定义了三个方法,配置了三个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.创建AnnotationConfigApplicationContext类型的Spring容器。
2.获取Spring容器里的Environment接口之后调用setActiveProfiles方法激活Profile
3.把Java配置类注册给Spring容器。
4.刷新Spring容器,完成Bean的创建。

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

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

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

它的参数是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:

在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.配置类AppConfigClassic带有@Profile(value = “classicProfile”)注解。只有激活了classicProfile,Spring容器才会创建该配置类里的Bean;否则不创建。
2.配置类AppConfigCountry带有@Profile(value = “countryProfile”)注解。只有激活了countryProfile,Spring容器才会创建该配置类里的Bean;否则不创建。
3.配置类AppConfig没带@Profile注解。不管激活了哪些Profile,Spring容器都会创建该配置类里的Bean

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

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

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

Original: https://www.cnblogs.com/evanlin/p/16069665.html
Author: 林雪波
Title: Profile

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/571386/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球