詳解Spring中的Environment外部化配置管理
Environment的中文意思是環(huán)境,它表示整個(gè)spring應(yīng)用運(yùn)行時(shí)的環(huán)境信息,它包含兩個(gè)關(guān)鍵因素
- profiles
- properties
profiles
profiles這個(gè)概念相信大家都已經(jīng)理解了,最常見的就是不同環(huán)境下,決定當(dāng)前spring容器中的不同配置上下文的解決方案。比如針對(duì)開發(fā)環(huán)境、測(cè)試環(huán)境、生產(chǎn)環(huán)境,構(gòu)建不同的application.properties配置項(xiàng),這個(gè)時(shí)候我們可以通過profiles這個(gè)屬性來決定當(dāng)前spring應(yīng)用上下文中生效的配置項(xiàng)。
實(shí)際上,通過profiles可以針對(duì)bean的配置進(jìn)行邏輯分組。 簡(jiǎn)單來說,我們可以通過profiles來針對(duì)不同的bean進(jìn)行邏輯分組,這個(gè)分組和bean本身的定義沒有任何關(guān)系,無論是xml還是注解方式,都可以配置bean屬于哪一個(gè)profile分組。
當(dāng)存在多個(gè)profile分組時(shí),我們可以指定哪一個(gè)profile生效,當(dāng)然如果不指定,spring會(huì)根據(jù)默認(rèn)的profile去執(zhí)行。我們來通過一個(gè)代碼演示一下。
ProfileService
創(chuàng)建一個(gè)普通的類,代碼如下
public class ProfileService { private String profile; public ProfileService(String profile) { this.profile = profile; } @Override public String toString() { return "ProfileService{" + "profile='" + profile + '\'' + '}'; } }
聲明一個(gè)配置類
在配置類中,構(gòu)建兩個(gè)bean,配置不同的profile。
@Configuration public class ProfileConfiguration { @Bean @Profile("dev") public ProfileService profileServiceDev(){ return new ProfileService("dev"); } @Bean @Profile("prod") public ProfileService profileServiceProd(){ return new ProfileService("prod"); } }
定義測(cè)試方法
public class ProfileMain { public static void main(String[] args) { AnnotationConfigApplicationContext applicationContext=new AnnotationConfigApplicationContext(); // applicationContext.getEnvironment().setActiveProfiles("prod"); applicationContext.register(ProfileConfiguration.class); applicationContext.refresh(); System.out.println(applicationContext.getBean(ProfileService.class)); } }
可以通過很多種方式來激活配置,默認(rèn)情況下不添加applicationContext.getEnvironment().setActiveProfiles("prod");
時(shí),會(huì)發(fā)現(xiàn)bean沒有被裝載。添加了之后,會(huì)根據(jù)當(dāng)前激活的profiles來決定裝載哪個(gè)bean。
除此之外,我們還可以在啟動(dòng)參數(shù)中增加-Dspring.profiles.active=prod
來決定當(dāng)前激活哪個(gè)profile。該屬性可以配置在系統(tǒng)環(huán)境變量、JVM系統(tǒng)屬性、等。
注意配置文件不是單選;可能會(huì)同時(shí)激活多個(gè)配置文件,編程式的使用方法setActiveProfiles(),該方法接收String數(shù)組參數(shù),也就是多個(gè)配置文件名
applicationContext.getEnvironment().setActiveProfiles("prod","dev");
如果沒有任何profile配置被激活,默認(rèn)的profile將會(huì)激活。 默認(rèn)profile配置文件可以更改,通過環(huán)境變量的setDefaultProfiles方法,或者是聲明的spring.profiles.default屬性值
profiles總結(jié)
簡(jiǎn)單總結(jié)一下profiles,通過profiles可以最一組bean進(jìn)行邏輯分組,這些邏輯分組的bean會(huì)根據(jù)Environment上下文中配置的激活的profile來進(jìn)行加載,也就是Environment對(duì)于profiles配置來說,它能決定當(dāng)前激活的是哪個(gè)profile配置。
- 一個(gè)profile就是一組Bean定義的邏輯分組。
- 這個(gè)分組,也就 這個(gè)profile,被賦予一個(gè)命名,就是這個(gè)profile名字。
- 只有當(dāng)一個(gè)profile處于active狀態(tài)時(shí),它對(duì)應(yīng)的邏輯上組織在一起的這些Bean定義才會(huì)被注冊(cè)到容器中。
- Bean添加到profile可以通過XML定義方式或者annotation注解方式。
- Environment對(duì)于profile所扮演的角色是用來指定哪些profile是當(dāng)前活躍的缺省。
Properties
properties的作用就是用來存放屬性的,它可以幫我們管理各種配置信息。這個(gè)配置的來源可以是properties文件、JVM properties、系統(tǒng)環(huán)境變量、或者專門的Properties對(duì)象等。
我們來看一下Environment這個(gè)接口,它繼承了PropertyResolver,這個(gè)接口和屬性的操作有關(guān),也就是我們可以通過Environment來設(shè)置和獲得相關(guān)屬性。
public interface Environment extends PropertyResolver { String[] getActiveProfiles(); String[] getDefaultProfiles(); /** @deprecated */ @Deprecated boolean acceptsProfiles(String... var1); boolean acceptsProfiles(Profiles var1); }
至此,我們可以可以簡(jiǎn)單的總結(jié)Environment的作用,Environment提供了不同的profile配置,而PropertyResolver提供了配置的操作,由此我們可以知道,Spring 容器可以根據(jù)不同的profile來獲取不同的配置信息,從而實(shí)現(xiàn)Spring容器中運(yùn)行時(shí)環(huán)境的處理。
environment的應(yīng)用
在spring boot應(yīng)用中,修改application.properties配置
env=default
創(chuàng)建一個(gè)Controller進(jìn)行測(cè)試
@RestController public class EnvironementController { @Autowired Environment environment; @GetMapping("/env") public String env(){ return environment.getProperty("env"); } }
指定profile屬性
在前面的內(nèi)容中我們介紹了profile和property這兩個(gè)概念,現(xiàn)在我們來結(jié)合使用加深對(duì)這兩者的理解。
在spring boot應(yīng)用中,默認(rèn)的外部化配置是application.properties文件,事實(shí)上,除了這個(gè)默認(rèn)的配置文件之外,我們還可以使用springboot中的約定命名格式來實(shí)現(xiàn)不同環(huán)境的配置
application-profile.properties
當(dāng)前spring boot應(yīng)用選擇使用哪個(gè)properties文件作為上下文環(huán)境配置,取決與當(dāng)前激活的profile。同樣,我們可以通過很多種方式來激活,比如在application.properties中增加spring.profiles.active=dev
這種方式,也可以在JVM參數(shù)中增加該配置來指定生效的配置。
在不指定的情況下,則使用默認(rèn)的配置文件,簡(jiǎn)單來說,如果沒有顯式激活某一個(gè)配置文件,那么應(yīng)用程序就將加載application-default.properties中的屬性。
這個(gè)功能非常實(shí)用,一般的公司里面都會(huì)有幾套運(yùn)行環(huán)境,比如開發(fā)、測(cè)試、生產(chǎn)環(huán)境,這些環(huán)境中會(huì)有一些配置信息是不同的,比如服務(wù)器地址。那我們需要針對(duì)不同的環(huán)境使用指定的配置信息,通過這種方式就可以很方便的去解決。
@Value注解的使用
在properties文件中定義的屬性,除了可以通過environment的getProperty方法獲取之外,spring還提供了@Value注解,
@RestController public class EnvironementController { @Value("${env}") private String env; @GetMapping("/env") public String env(){ return env; } }
spring容器在加載一個(gè)bean時(shí),當(dāng)發(fā)現(xiàn)這個(gè)Bean中有@Value注解時(shí),那么它可以從Environment中將屬性值進(jìn)行注入,如果Environment中沒有這個(gè)屬性,則會(huì)報(bào)錯(cuò)。
Spring Environment原理設(shè)計(jì)
結(jié)合前面咱們講過的內(nèi)容,我們來推測(cè)一下Environment的實(shí)現(xiàn)原理。
簡(jiǎn)單演示一下Environment中的配置來源
- @Value("${java.version}") 獲取System.getProperties , 獲取系統(tǒng)屬性
- 配置command的jvm參數(shù),
-Denvtest=command
基于現(xiàn)有的內(nèi)容的推導(dǎo),我們可以畫出下面這樣一個(gè)圖。
- 第一部分是屬性定義,這個(gè)屬性定義可以來自于很多地方,比如application.properties、或者系統(tǒng)環(huán)境變量等。
- 然后根據(jù)約定的方式去指定路徑或者指定范圍去加載這些配置,保存到內(nèi)存中。
- 最后,我們可以根據(jù)指定的key從緩存中去查找這個(gè)值。
下面這個(gè)是表示Environment的類關(guān)系圖,這個(gè)類關(guān)系圖還是非常清晰的體現(xiàn)了Environment的原理。
上述類圖的核心API說明如下
Environment接口,繼承了PropertyResolver。 PropertyResolver,它主要有兩個(gè)作用。
- 通過
propertyName
屬性名獲取與之對(duì)應(yīng)的propertValue
屬性值(getProperty)。 - 把
${propertyName:defaultValue}
格式的屬性占位符,替換為實(shí)際的值(resolvePlaceholders)。
PropertyResolver的具體實(shí)現(xiàn)類是PropertySourcesPropertyResolver,屬性源的解決方案。該類是體系中唯一的完整實(shí)現(xiàn)類。它以PropertySources屬性源集合(內(nèi)部持有屬性源列表List )為屬性值的來源,按序遍歷每個(gè)PropertySource,獲取到一個(gè)非null的屬性值則返回。
其中,PropertySourcesPropertyResolver中的List ,表示不同屬性源的來源,它的類關(guān)系圖如下,表示針對(duì)不同數(shù)據(jù)源的存儲(chǔ)。
到此這篇關(guān)于詳解Spring中的Environment外部化配置管理的文章就介紹到這了,更多相關(guān)Spring Environment外部化配置內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringBoot?Test的webEnvironment源碼解讀
- springboot的EnvironmentPostProcessor接口方法源碼解析
- Spring運(yùn)行環(huán)境Environment的解析
- Spring?Boot讀取配置文件內(nèi)容的3種方式(@Value、Environment和@ConfigurationProperties)
- Spring之底層架構(gòu)核心概念Environment及用法詳解
- SpringBoot擴(kuò)展點(diǎn)EnvironmentPostProcessor實(shí)例詳解
- 基于Spring Boot的Environment源碼理解實(shí)現(xiàn)分散配置詳解
- Spring之Environment類的使用方式
相關(guān)文章
IntelliJ IDEA設(shè)置顯示內(nèi)存指示器和設(shè)置內(nèi)存大小的方法
這篇文章主要介紹了IntelliJ IDEA設(shè)置顯示內(nèi)存指示器和設(shè)置內(nèi)存大小的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04mybatis實(shí)現(xiàn)對(duì)數(shù)據(jù)的增刪查改實(shí)例詳解
這篇文章主要介紹了mybatis實(shí)現(xiàn)對(duì)數(shù)據(jù)的增刪查改實(shí)例詳解的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-07-07SpringBoot?Schedule調(diào)度任務(wù)的動(dòng)態(tài)管理
Scheduled定時(shí)任務(wù)是Spring?boot自身提供的功能,所以不需要引入Maven依賴包,下面這篇文章主要給大家介紹了關(guān)于SpringBoot通過@Scheduled實(shí)現(xiàn)定時(shí)任務(wù)以及問題解決的相關(guān)資料,需要的朋友可以參考下2023-02-02java多線程Thread的實(shí)現(xiàn)方法代碼詳解
這篇文章主要介紹了java多線程Thread的實(shí)現(xiàn)方法代碼詳解,涉及start(),run(),stop(),interrupt(),isInterrupted(),join()和join(long millis)等方法的介紹,具有一定借鑒價(jià)值,需要的朋友可以了解下。2017-11-11SpringBoot中@ComponentScan的使用詳解
這篇文章主要介紹了SpringBoot中@ComponentScan的使用詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11