一般服務端應用開發須要通過如下幾個流程:java
開發 -> 測試 -> RC驗證 -> 上線web
這就涉及到四個不一樣的環境,開發環境、測試環境、RC環境以及生產環境,爲了不不一樣環境之間相互干擾,一般須要獨立部署數據庫、緩存服務器等,那麼應用配置也要作相應的調整。spring
爲了解決不一樣環境配置切換問題,不少人的作法是:把配置文件根據不一樣的環境,放到不一樣的目錄或文件中,打包時經過gradle或maven,經過命令行參數指定要打哪一個環境的包。這樣就能夠針對不一樣的環境生成不一樣的包。但這樣的作法有如下幾個問題:docker
gradle或maven打包腳本文件須要重複編寫「選擇文件」打包的邏輯,增長不少重複勞動的成功;數據庫
在jenkins等集成環境中,須要針對每一個應用,不一樣的環境作相應的設置;bootstrap
須要管理不一樣環境的包,帶來的成本;部署時,須要注意包與運行環境是否一致;緩存
若是運行在docker中時,由於包不一樣因此要針對不一樣的環境,構建相應的鏡像。tomcat
這時也許有人會說,把配置都從包裏剝離出來,放到配置中心就能夠了,可是不一樣環境對應的配置中心地址也是不同的。安全
Spring中的Profile 是什麼?bash
Spring中的Profile功能其實早在Spring 3.1的版本就已經出來,它能夠理解爲咱們在Spring容器中所定義的Bean的邏輯組名稱,只有當這些Profile被激活的時候,纔會將Profile中所對應的Bean註冊到Spring容器中。
舉個更具體的例子,咱們之前所定義的Bean,當Spring容器一啓動的時候,就會一股腦的所有加載這些信息完成對Bean的建立;而使用了Profile以後,它會將Bean的定義進行更細粒度的劃分,將這些定義的Bean劃分爲幾個不一樣的組,當Spring容器加載配置信息的時候,首先查找激活的Profile,而後只會去加載被激活的組中所定義的Bean信息,而不被激活的Profile中所定義的Bean定義信息是不會加載用於建立Bean的。
爲了使用不一樣的環境,咱們首先對不一樣的環境,定義相應的profile名稱。
好比,開發環境的profile爲:dev;測試環境的profile爲:test;RC環境的profile爲:rc;生產環境的profile爲:prod。
下面舉個dubbo不一樣環境下,使用不一樣配置的方法:
上面例子中,當激活相應的profile時,相應的配置文件纔會導入。
好比:profile爲dev時,導入dubbo-dev.properties。
注意:全部spring xml schema的版本必須是4.0以上,好比:http://www.springframework.org/schema/util/spring-util-4.3.xsd。spring 默認profile爲default, 在沒有指定profile的,會被默認爲default。
若是咱們使用配置中心的話,上面的配置還能夠更簡單。等配置中心投產後咱們再討論。
Spring boot中使用profile切換配置
Spring boot中默認加載的配置文件是:application.properties或application.yml。當激活profile後(後面咱們討論如何激活profile),能夠經過profile自動選擇加載的application-{profile}.properties或application-{profile}.yml格式的配置文件。
好比:profile爲dev時,會加載application.properties或application.yml外,還會加載application-dev.properties或application-dev.yml配置。
另外若是引入Spring cloud 時,也會加載啓動配置bootstrap.properties或bootstrap.yml以及bootstrap-{profile}.properties 或 bootstrap-{profile}.yml。
因此把各個環境公共的配置寫在application.properties或application.yml中。把不一樣環境的配置寫在application-{profile}.properties或application-{profile}.yml中。
@Profile註解的使用
使用java進行配置時,能夠經過@Profile註解,實現不一樣環境使用配置策略。好比swagger如今使用很廣泛了,可是它存在必定的安全問題,若是生產環境中也暴露swagger的話,風險仍是比較大的,建議只在開發環境和測試環境啓用,配置例子以下:
將上面的代碼保存到logback-spring.xml文件中,而不是logback.xml中。
logback中profile的使用
在開發環境或測試環境中,爲了方便排查問題,咱們會使用DEBUG甚至TRACE級別的日誌,而在生產環境中,避免日誌增加過快,儘可能只是輸出ERROR級別的日誌。這就須要日誌配置也要能根據不一樣的環境,使用不一樣的配置策略。
spring boot中的logback就能夠知足這樣的需求,例子以下:
將上面的代碼保存到logback-spring.xml文件中,而不是logback.xml中。
Spring boot 激活 profile的幾種方式
在配置文件中直接指定
spring.profiles.active=test
這種方式很是不靈活,在實際開發部不太會使用到
使用佔位符
在打包時替換,以mavne爲例:
首先在配置文件中增長:
spring.profiles.active=@package.target@
在pom.xml中增長不一樣環境打包的配置:
執行打包命令:
mvn package -Ptest
缺點:每次打包都要指定profile
JVM參數方式
java命令行指定:
java -jar app.jar --spring.profiles.active=dev
tomcat 中 catalina.bat(.sh中不用「set」) 添加JAVA_OPS。經過設置active選擇不一樣配置文件:
set JAVA_OPTS="-Dspring.profiles.active=test"
eclipse 中啓動tomcat。項目右鍵 run as –> run configuration–>Arguments–> VM arguments中添加。
-Dspring.profiles.active="dev"
在微服務的時代,會不會以爲有點麻煩呢?
web.xml方式
<init-param>
<param-name>spring.profiles.active</param-name>
<param-value>prod</param-value>
</init-param>
標註方式(junit單元測試很是實用)
@ActiveProfiles({"dev"})
ENV方式(建議使用此方式)
設置系統環境變量:SPRING_PROFILES_ACTIVE(注意:是大寫)
好比mac開發環境中設置環境變量的方法:
vi ~/.bash_profile
在~/.bash_profile中增長以下內容:
export SPRING_PROFILES_ACTIVE=dev
注意: mac eclipse中是獲取不到環境變量的解決辦法,參考這文章進行處理,http://blog.csdn.net/zhzdeng/article/details/64921967
總結
上面關於profile的東西,基本能知足工做的須要了。使用profile後,能夠減化因不一樣環境配置差別,而帶來的配置管理以及打包工做。
儘可能使用環境變量來激活profile,若是是可執行的包,也可使用java命令行指定,其它方式不建議使用。
使用profile後,使得應用能更容易接入配置中心,以及使用docker容器技術,因此很是有意義。
——————————————————分割線——————————————————
我是黑少,直男一枚,微服務硬核玩家,喜歡分享、愛交友人、崇尚「實踐出真知」的理念,以折騰鼓搗代碼爲樂
個人微信:weiweiweiblack (備註:開源中國 )
微信公號:黑少微服務,專一微服務技術分享,非技術不八卦!