軟件開發的一般流程為工程師開發 -> 測試 -> 上線,因此就涉及到三個不同的環境,開發環境、測試環境以及生產環境,通常這三個環境會有很多配置參數不同,例如數據源、文件路徑、url等,如果每次上線一個新版本時都手動修改配置會十分繁瑣,容易出錯。spring 為我們提供了 profile 機制來解決這個問題。
spring允許我們通過定義 profile 來將若干不同的 bean 定義組織起來,從而實現不同環境自動激活不同的 profile 來切換配置參數的功能,下面介紹以 xml 的方式定義 profile、如何激活 profile以及定義默認的 profile,整個過程我以配置不同環境的數據源為例,為了簡化配置,這里假設只有開發和生產兩個環境。
數據源定義為
- <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
- <property name="user" value="${jdbc.user}" />
- <property name="password" value="${jdbc.password}" />
- <property name="jdbcUrl" value="${jdbc.jdbcUrl}" />
- <property name="driverClass" value="${jdbc.driverClass}" />
- <property name="initialPoolSize" value="${c3p0.initialPoolSize}"/>
- <property name="acquireIncrement" value="${c3p0.acquireIncrement}"/>
- <property name="minPoolSize" value="${c3p0.minPoolSize}"/>
- <property name="maxIdleTime" value="${c3p0.maxIdleTime}"/>
- <property name="idleConnectionTestPeriod" value="${c3p0.idleConnectionTestPeriod}" />
- <property name="preferredTestQuery" value="${c3p0.preferredTestQuery}"/>
- </bean>
classpath下外部資源文件有兩個 settings-development.properties 和 settings-production.properties,分別是開發環境和生產環境的數據源配置參數,內容如下
settings-development.properties
- jdbc.user=root
- jdbc.password=111111
- jdbc.driverClass=com.mysql.jdbc.Driver
- jdbc.jdbcUrl=jdbc:mysql://localhost:3306/xxx
- c3p0.minPoolSize=5
- c3p0.initialPoolSize=5
- c3p0.acquireIncrement=5
- c3p0.maxIdleTime=3600
- c3p0.idleConnectionTestPeriod=3600
- c3p0.preferredTestQuery=select 1
settings-production.properties
- jdbc.user=xxx
- jdbc.password=xxxx
- jdbc.driverClass=com.mysql.jdbc.Driver
- jdbc.jdbcUrl=jdbc:mysql:///xxx
- c3p0.minPoolSize=20
- c3p0.initialPoolSize=20
- c3p0.acquireIncrement=10
- c3p0.maxIdleTime=3600
- c3p0.idleConnectionTestPeriod=3600
- c3p0.preferredTestQuery=select 1
1. 定義 profile
現在就可以通過定義 profile 來將開發和生產環境的數據源配置分開,這里我們定義兩個 profile,一個名稱是 development,另一個名稱是 production
- <!-- 開發環境配置文件 -->
- <beans profile="development">
- <context:property-placeholder location="classpath:settings-development.properties"/>
- </beans>
- <!-- 生產環境配置文件 -->
- <beans profile="production">
- <context:property-placeholder location="classpath:settings-production.properties"/>
- </beans>
2. 定義默認 profile
默認 profile 是指在沒有任何 profile 被激活的情況下,默認 profile 內定義的內容將被使用,通常可以在 web.xml 中定義全局 servlet 上下文參數 spring.profiles.default 實現,代碼如下
- <!-- 配置spring的默認profile -->
- <context-param>
- <param-name>spring.profiles.default</param-name>
- <param-value>development</param-value>
- </context-param>
3. 激活 profile
spring 為我們提供了大量的激活 profile 的方法,可以通過代碼來激活,也可以通過系統環境變量、JVM參數、servlet上下文參數來定義 spring.profiles.active 參數激活 profile,這里我們通過定義 JVM 參數實現。
在生產環境中,以 tomcat 為例,我們在 tomcat 的啟動腳本中加入以下 JVM 參數
- -Dspring.profiles.active="production"
而開發環境中不需要定義該參數,如果不定義,則會使用我們指定的默認 profile ,在這里也就是名稱為 development 的 profile。這樣當項目部署到不同的環境時,便可以通過 JVM 參數來實現不同環境 profile 自動激活。
4. 延伸
該參數還可以延伸,以至於我們可以在 java 代碼中繼續通過該參數來區分不同的環境,從而執行不同的操作,代碼如下
public class Config {
public static String ENV = "development";
}
public class InitConfigListener implements ServletContextListener {
public void contextInitialized(ServletContextEvent sce) {
//偵測jvm環境,並緩存到全局變量中
String env = System.getProperty("spring.profiles.active");
if(env == null) {
Config.ENV = "development";
} else {
Config.ENV = env;
}
}
}
在項目初始化時獲取到該參數后,我們便可以在項目任何位置使用,來執行一些不同環境需要區別對待的操作,例如統計流量的代碼只需要在生產環境激活,就可以在jsp中加入如下代碼
<!-- 生產環境統計、推送代碼 -->
<c:if test="${env == 'production' }">
<script>
//統計代碼
..
</script>
</c:if>