一般在一個(gè)項(xiàng)目中,總是會(huì)有好多個(gè)環(huán)境。比如:
每個(gè)環(huán)境上的配置文件總是不一樣的,甚至開發(fā)環(huán)境中每個(gè)開發(fā)者的環(huán)境可能也會(huì)有一點(diǎn)不同,配置讀取可是一個(gè)讓人有點(diǎn)傷腦筋的問題。 Spring Boot提供了一種優(yōu)先級配置讀取的機(jī)制來幫助我們從這種困境中走出來。 常規(guī)情況下,我們都知道Spring Boot的配置會(huì)從 外部化的配置在應(yīng)用中管理配置并不是一個(gè)容易的任務(wù),尤其是在應(yīng)用需要部署到多個(gè)環(huán)境中時(shí)。通常會(huì)需要為每個(gè)環(huán)境提供一個(gè)對應(yīng)的屬性文件,用來配置各自的數(shù)據(jù)庫連接信息、服務(wù)器信息和第三方服務(wù)賬號等。通常的應(yīng)用部署會(huì)包含開發(fā)、測試和生產(chǎn)等若干個(gè)環(huán)境。不同的環(huán)境之間的配置存在覆蓋關(guān)系。測試環(huán)境中的配置會(huì)覆蓋開發(fā)環(huán)境,而生產(chǎn)環(huán)境中的配置會(huì)覆蓋測試環(huán)境。Spring 框架本身提供了多種的方式來管理配置屬性文件。Spring 3.1 之前可以使用 PropertyPlaceholderConfigurer。 Spring Boot 所提供的配置優(yōu)先級順序比較復(fù)雜。按照優(yōu)先級從高到低的順序,具體的列表如下所示。
Spring Boot 的這個(gè)配置優(yōu)先級看似復(fù)雜,其實(shí)是很合理的。比如命令行參數(shù)的優(yōu)先級被設(shè)置為最高。 SpringApplication 類默認(rèn)會(huì)把以“--”開頭的命令行參數(shù)轉(zhuǎn)化成應(yīng)用中可以使用的配置參數(shù),如 “--name=Alex” 會(huì)設(shè)置配置參數(shù) “name” 的值為 “Alex”。如果不需要這個(gè)功能,可以通過 “SpringApplication.setAddCommandLineProperties(false)” 禁用解析命令行參數(shù)。 RandomValuePropertySource 可以用來生成測試所需要的各種不同類型的隨機(jī)值,從而免去了在代碼中生成的麻煩。RandomValuePropertySource 可以生成數(shù)字和字符串。數(shù)字的類型包含 int 和 long,可以限定數(shù)字的大小范圍。以“random.”作為前綴的配置屬性名稱由 RandomValuePropertySource 來生成,如代碼清單 3 所示。 清單 3. 使用 RandomValuePropertySource 生成的配置屬性user.id=${random.value} user.count=${random.int} user.max=${random.long} user.number=${random.int(100)} user.range=${random.int[100, 1000]} 屬性文件 屬性文件是最常見的管理配置屬性的方式。Spring Boot 提供的 SpringApplication 類會(huì)搜索并加載 application.properties 文件來獲取配置屬性值。SpringApplication 類會(huì)在下面位置搜索該文件。
上面的順序也表示了該位置上包含的屬性文件的優(yōu)先級。優(yōu)先級按照從高到低的順序排列。可以通過“spring.config.name”配置屬性來指定不同的屬性文件名稱。也可以通過“spring.config.location”來添加額外的屬性文件的搜索路徑。如果應(yīng)用中包含多個(gè) profile,可以為每個(gè) profile 定義各自的屬性文件,按照“application-{profile}”來命名。 對于配置屬性,可以在代碼中通過“@Value”來使用,如代碼清單 4 所示。 清單 4. 通過“@Value”來使用配置屬性@RestController @EnableAutoConfiguration public class Application { @Value("${name}") private String name; @RequestMapping("/") String home() { return String.format("Hello %s!", name); } } 在代碼清單 4 中,變量 name 的值來自配置屬性中的“name”屬性。 YAML相對于屬性文件來說,YAML 是一個(gè)更好的配置文件格式。YAML 在 Ruby on Rails 中得到了很好的應(yīng)用。SpringApplication 類也提供了對 YAML 配置文件的支持,只需要添加對 SnakeYAML 的依賴即可。代碼清單 5 給出了 application.yml 文件的示例。 清單 5. 使用 YAML 表示的配置屬性spring: profiles: development db: url: jdbc:hsqldb:file:testdb username: sa password: --- spring: profiles: test db: url: jdbc:mysql://localhost/test username: test password: test 代碼清單 5 中的 YAML 文件同時(shí)給出了 development 和 test 兩個(gè)不同的 profile 的配置信息,這也是 YAML 文件相對于屬性文件的優(yōu)勢之一。 清單 6. 使用 YAML 文件的 Java 類@Component @ConfigurationProperties(prefix="db") public class DBSettings { private String url; private String username; private String password; } http://www.ibm.com/developerworks/cn/java/j-lo-spring-boot/index.html
這意味著,如果Spring Boot在優(yōu)先級更高的位置找到了配置,那么它就會(huì)無視優(yōu)先級低的配置。 比如,我在 db.jdbc.driver=com.mysql.jdbc.Driver db.jdbc.url=jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&characterEncoding=UTF-8 db.jdbc.username=username db.jdbc.password=password 在自己項(xiàng)目調(diào)試的階段,項(xiàng)目總是會(huì)使用本地的MySQL數(shù)據(jù)庫。而一旦打包之后,在外部聲明一個(gè) 啟動(dòng)Jar包的時(shí)候, 指定一個(gè)外部配置文件:
這樣一來,我們在開發(fā)者的機(jī)器上總是使用自己的配置,而一到對應(yīng)的環(huán)境,就會(huì)使用高級的位置所做的配置。 在代碼中讀取這些配置也是非常方便的,在代碼的邏輯中,其實(shí)是無需去關(guān)心這個(gè)配置是從什么地方來的,只用關(guān)注能獲取什么配置就夠了。 public class ApplicationConfigure { @Value("${db.jdbc.driver}") private String jdbcDriver; @Value("${db.jdbc.url}") private String jdbcUrl; @Value("${db.jdbc.username}") private String jdbcUsername; @Value("${db.jdbc.password}") private String jdbcPassword; // mysql config class // ..... }
有時(shí)候我們在項(xiàng)目啟動(dòng)的時(shí)候,總是需要先啟動(dòng)一些初始化的類,以前比較常見的做法是寫再 public class ApplicationConfigure implements CommandLineRunner { @Value("${db.jdbc.driver}") private String jdbcDriver; @Value("${db.jdbc.url}") private String jdbcUrl; @Value("${db.jdbc.username}") private String jdbcUsername; @Value("${db.jdbc.password}") private String jdbcPassword; // mysql config class // ..... @Override public void run(String... strings) throws Exception { // 預(yù)先加載的一些方法,類,屬性。 } }
如果有多個(gè) 通過這一系列API的幫助,Spring Boot讓環(huán)境配置變得輕松很多。 http://www.cnblogs.com/whthomas/p/5270917.html http://www.ibm.com/developerworks/cn/java/j-lo-spring-boot/index.html Tomcat # tomcat最大線程數(shù),默認(rèn)為200 server.tomcat.max-threads=800 # tomcat的URI編碼 server.tomcat.uri-encoding=UTF-8 # 存放Tomcat的日志、Dump等文件的臨時(shí)文件夾,默認(rèn)為系統(tǒng)的tmp文件夾(如:C:\Users\Shanhy\AppData\Local\Temp) server.tomcat.basedir=H:/springboot-tomcat-tmp # 打開Tomcat的Access日志,并可以設(shè)置日志格式的方法: #server.tomcat.access-log-enabled=true #server.tomcat.access-log-pattern= # accesslog目錄,默認(rèn)在basedir/logs #server.tomcat.accesslog.directory= # 日志文件目錄 logging.path=H:/springboot-tomcat-tmp # 日志文件名稱,默認(rèn)為spring.log logging.file=myapp.log Jetty 如果你要選擇Jetty,也非常簡單,就是把pom中的tomcat依賴排除,并加入Jetty容器的依賴,如下: <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </exclusion> </exclusions> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jetty</artifactId> </dependency> <dependencies> 打包 打包方法: E:\spring-boot-sample>mvn clean package
可以追加參數(shù) -Dmaven.test.skip=true (-DskipTests)跳過測試。 打包后的文件存放于項(xiàng)目下的target目錄中,如:spring-boot-sample-0.0.1-SNAPSHOT.jar 二、部署到JavaEE容器
<!-- <packaging>jar</packaging> --> <packaging>war</packaging>
<dependency> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </exclusion> </exclusions> </dependency>
三、使用Profile區(qū)分環(huán)境spring boot 可以在 “配置文件”、“Java代碼類”、“日志配置” 中來配置profile區(qū)分不同環(huán)境執(zhí)行不同的結(jié)果 1、配置文件 app.name=MyApp server.port=8080 spring.profiles.active=dev application-dev.properties server.port=8081 application-stg.properties server.port=8082 在啟動(dòng)程序的時(shí)候通過添加 –spring.profiles.active={profile} 來指定具體使用的配置 例如我們執(zhí)行 java -jar demo.jar –spring.profiles.active=dev 那么上面3個(gè)文件中的內(nèi)容將被如何應(yīng)用? app.name 只存在于默認(rèn)配置文件 application.properties 中,因?yàn)橹付ōh(huán)境中不存在同樣的配置,所以該值不會(huì)被覆蓋
server.port 默認(rèn)為8080,但是我們指定了環(huán)境后,將會(huì)被覆蓋。如果指定stg環(huán)境,server.port 則為 8082
spring.profiles.active 默認(rèn)指定dev環(huán)境,如果我們在運(yùn)行時(shí)指定 –spring.profiles.active=stg 那么將應(yīng)用stg環(huán)境,最終 server.port 的值為8082
2、Java類中@Profile注解 // Dev 環(huán)境實(shí)現(xiàn)類 @Component @Profile("dev") public class DevSendMessage implements SendMessage { @Override public void send() { System.out.println(">>>>>>>>Dev Send()<<<<<<<<"); } }
// Stg環(huán)境實(shí)現(xiàn)類 @Component @Profile("stg") public class StgSendMessage implements SendMessage { @Override public void send() { System.out.println(">>>>>>>>Stg Send()<<<<<<<<"); } } // 啟動(dòng)類 @SpringBootApplication public class ProfiledemoApplication { @Value("${app.name}") private String name; @Autowired private SendMessage sendMessage; @PostConstruct public void init(){ sendMessage.send();// 會(huì)根據(jù)profile指定的環(huán)境實(shí)例化對應(yīng)的類 } } 3、logback-spring.xml也支持有節(jié)點(diǎn)來支持區(qū)分 <?xml version="1.0" encoding="UTF-8"?> <configuration> <include resource="org/springframework/boot/logging/logback/base.xml" /> <logger name="org.springframework.web" level="INFO"/> <springProfile name="default"> <logger name="org.springboot.sample" level="TRACE" /> </springProfile> <springProfile name="dev"> <logger name="org.springboot.sample" level="DEBUG" /> </springProfile> <springProfile name="staging"> <logger name="org.springboot.sample" level="INFO" /> </springProfile> </configuration> 再說一遍文件名不要用logback.xml 請使用logback-spring.xml 四、指定外部的配置文件有些系統(tǒng),關(guān)于一些數(shù)據(jù)庫或其他第三方賬戶等信息,由于安全問題,其配置并不會(huì)提前配置在項(xiàng)目中暴露給開發(fā)人員。 java -jar demo.jar --spring.config.location=/opt/config/application.properties 其中文件名隨便定義,無固定要求。 五、創(chuàng)建一個(gè)Linux 應(yīng)用的sh腳本下面幾個(gè)腳本僅供參考,請根據(jù)自己需要做調(diào)整 #!/bin/sh rm -f tpid nohup java -jar /data/app/myapp.jar --spring.profiles.active=stg > /dev/null 2>&1 & echo $! > tpid
stop.sh tpid=`cat tpid | awk '{print $1}'` tpid=`ps -aef | grep $tpid | awk '{print $2}' |grep $tpid` if [ ${tpid} ]; then kill -9 $tpid fi check.sh tpid=`cat tpid | awk '{print $1}'` tpid=`ps -aef | grep $tpid | awk '{print $2}' |grep $tpid` if [ ${tpid} ]; then echo App is running. else echo App is NOT running. fi kill.sh #!/bin/sh # kill -9 `ps -ef|grep 項(xiàng)目名稱|awk '{print $2}'` kill -9 `ps -ef|grep demo|awk '{print $2}'` http://www.cnblogs.com/duyinqiang/p/5696342.html |
|