Spring Boot 中提供一個全域性的組態檔:application.properties
,這個組態檔的作用就是,允許我們通過這個組態檔去修改 Spring Boot 自動設定的預設值。
Spring Boot 支援兩種格式的組態檔:application.properties
和 application.yml
。
yml 等同於 yaml
,寫法看個人喜歡,我喜歡寫成 application.yml
application.properties
和 application.yml
,它們的區別在於語法不同,但本質上是一樣的。application.properties
使用鍵值對的方式來設定,而 application.yml
使用縮排和冒號的方式來設定。
properties 作為字尾的組態檔,語法是這樣的:key = value
,如果有多級設定項,則是 first.second.third = value
。
key=value
first.second.third=value
範例:
key=value
game.name=GTA5
這裡的 key
和 game.name
都是屬性名稱,而 value
和 GTA5
是屬性的值。
game.list=GTA5,NBA2K,AC
這裡的 game.list
這個列表包含了 3 個元素。
game.map.key1=value1
game.map.key2=value2
這裡的 game.map
是一個 Map,這個 Map 包含了兩個元素,key1 對映到 value1,key2 對映到 value2
game.name=GTA5
# 參照上面已定義的屬性
great.game=${game.name}
yml 作為字尾的組態檔,語法是這樣的:key: value
。使用冒號代替等號,同時冒號後面需要跟上一個空格符,不可省略。
key: value
first:
second:
third: value
範例:
key: value
game:
name: GTA5
game:
list:
- GTA5
- NBA2K
- AC
game:
map:
key1: value1
key2: value2
game:
name: GTA5
great:
game: @{game.name}
一般專案中在不同環境下都有不同的設定,還是以這個 Tomcat 的埠號為例:
目前有 3 個環境,分別是開發環境、測試環境、生產環境。在開發環境下,埠號是 4790;測試環境下,埠號是 4791;生產環境下是 4792。
application-dev.yml
server:
port: 4790
application-test.yml
server:
port: 4791
application-prod.yml
server:
port: 4792
現在,通過 spring.profiles.active
這個設定項,在 application.yml
中指定我們想要切換的組態檔,現在指定使用開發環境的組態檔:
# 指定使用 application-dev.yml 這個組態檔
spring:
profiles:
active: dev
啟動 Spring Boot 應用,控制檯輸出:
2023-03-16 15:41:48.122 INFO 3356 --- [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat started on port(s): 4790 (http) with context path ''
指定使用測試環境下的組態檔:
# 指定使用 application-test.yml 這個組態檔
spring:
profiles:
active: test
啟動 Spring Boot 應用,控制檯輸出:
2023-03-16 15:42:21.462 INFO 24548 --- [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat started on port(s): 4791 (http) with context path ''
同理,指定使用生產環境的組態檔也是一樣的做法。
在 Spring Boot 專案中,自定義設定是經常用到的,我目前體會到的自定義的設定的作用有一點:設定與程式碼解耦
下面,我們看看如何自定義設定,並使用自己自定義設定的值:
server:
port: 4790
# 自定義的設定
demo:
author: god23bin
description: 點個免費的贊,我能開心好久!
上面自定義了兩個設定項,分別是 demo.author
和 demo.description
,接著,如何在程式碼中使用這些設定好的值呢?
目前使用這些設定好的值(讀取這些值),有以下幾種方式:
@Value
@ConfigurationProperties
我們寫一個普通的 Java 類,使用 Spring 提供的 @Value
來讀取這兩個值:
@Data
@Component
public class DemoCustomConfig {
/**
* 通過 @Value 註解讀取組態檔中的自定義設定項的值,使用 ${} 進行讀取
**/
@Value("${demo.author}")
private String author;
@Value("${demo.description}")
private String description;
}
上面的程式碼中,我在類上使用了 @Data
和 @Component
,@Data
是來自 Lombok 的,用於生成 getter 和 setter 方法,@Component
則將該類的範例物件交給 Spring 管理,接著在該類的兩個屬性上分別使用了 @Value
註解,通過 ${}
指定了我們要讀取的設定項。
進行測試,我們寫一個 Controller 判斷我們的讀取是否成功:
@RequestMapping("/demo")
@RestController
public class DemoController {
@Autowired
private DemoCustomConfig demoCustomConfig;
@GetMapping("/getCustomValue")
public ResponseEntity<String> getCustomValue() {
return ResponseEntity.ok(demoCustomConfig.getAuthor() + "說:" + demoCustomConfig.getDescription());
}
}
存取該介面:localhost:4790/demo/getCustomValue
@ConfigurationProperties
註解,它可以將組態檔中的的值繫結到 Java Bean 中,也就是通過這個 Bean 可以讀取到組態檔中設定的值,我們看看如何操作。
我們自定義一個用於讀取組態檔中設定項的類:
@Data
@Component
@ConfigurationProperties("system.demo")
public class SystemCustomConfig {
private String name;
private String version;
}
上面的程式碼,主要使用了 @ConfigurationProperties
這個註解,並指定了字首 system.demo
,同時這個類有兩個屬性,name 和 version, 這樣就相當於我們自定義了 system.demo.name
和 system.demo.version
這兩個屬性。
接著,我們就能在組態檔中寫這兩個我們自定義的設定項了:
server:
port: 4790
# 自定義的設定
system:
demo:
name: 超級系統
version: 1.0
寫完這裡的設定項,並不需要使用 @Value
去讀取,因為使用了 @ConfigurationProperties
註解,Spring 已經幫我們搞定了設定的值的讀取,至於它的實現原理,這裡先不深究。
進行測試,依然通過寫一個介面來測試我們通過:
@RequestMapping("/demo")
@RestController
public class DemoController {
@Autowired
private SystemCustomConfig systemCustomConfig;
@GetMapping("/getSystemVersion")
public ResponseEntity<String> getSystemVersion() {
return ResponseEntity.ok(systemCustomConfig.getName() + "版本:" + systemCustomConfig.getVersion());
}
}
存取該介面:localhost:4790/demo/getSystemVersion
不過,目前有個問題就是,我們自己寫了個 Java Bean 後,在組態檔中寫設定項的時候並沒有相關提示,這個就比較不友好,如果當我們自己寫的設定想要給其他人用的話,別人都不知道有什麼設定可以配。所以想要能像 Spring Boot 提供的設定提示一樣的話,就需要引入下面的依賴:
<!-- 組態檔處理器,組態檔進行繫結就會有提示 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>
spring-boot-configuration-processor
是一個用於生成設定後設資料的註解處理器。它會掃描使用@ConfigurationProperties
註解的類和方法,來獲取設定引數並生成設定後設資料。生成的設定後設資料可以用於 IDE 的自動補全和提示功能。
如果你引入了這個依賴並且重啟該 Spring Boot 專案後,依舊沒有提示的話,嘗試開啟 IDEA 中的 annotaion processing
。
對於我們自定義的設定,會出現這樣的提示:Cannot resolve configuration property(無法處理自定義的設定屬性),如下:
解決方式就是定義一下後設資料,用後設資料來描述這個屬性。當我們移動到這個設定項上時,出現提示,我們直接點選 Define configuration key xxx
就可以幫我們生成一個關於設定的後設資料檔案。
additional-spring-configuration-metadata.json
:
{
"properties": [
{
"name": "demo.author",
"type": "java.lang.String",
"description": "Demo的作者"
},
{
"name": "demo.description",
"type": "java.lang.String",
"description": "Demo的描述"
}
]
}
此時,還是一樣,重啟專案,如果黃色提示還是沒有去除的話,這裡建議重新用 Maven 進行一次 clean,接著重新編譯整個專案,就可以了。於此同時,也具有了自動提示功能。
上面說的是自定義的設定,現在這裡說自定義的組態檔,我們知道 Spring Boot 預設提供 application.properties
這個組態檔。那現在我們想自己寫一個組態檔,並且能在應用中讀取這個組態檔的資訊,該如何做呢?這裡就涉及到 @PropertySource
這個註解了。
自定義的組態檔:
custom.yml
:
# 自定義的組態檔
version: 2.0
description: 求關注!
讀取該組態檔的設定類:
@Data
@Configuration
@PropertySource("classpath:custom.yml")
public class DemoPropertiesSourceConfig {
@Value("${version}")
private String version;
@Value("${description}")
private String description;
}
這樣,就能讀取到自己編寫的組態檔的設定資訊了。
如果有多個自定義的組態檔,那麼可以使用 @PropertySources
註解,可以看到最後面多加了一個 s
,說明這個單詞是複數,通俗易懂。
@Configuration
@PropertySources({
@PropertySource("classpath:custom1.yml"),
@PropertySource("classpath:custom2.properties")
})
public class MyConfig {
// ...
}
組態檔的兩種寫法:properties 和 yml
專案中存在多個組態檔,可以使用 spring.profiles.active
屬性來切換使用哪個組態檔。
自定義的一些設定屬性(設定項),如何讀取呢?可以在程式中通過 @Value
或者@ConfigurationProperties
來讀取。
自定義的組態檔,可以通過 @PropertySource
來指定獲取該自定義組態檔的資訊。
希望各位螢幕前的靚仔靚女們
給個三連!你輕輕地點了個贊,那將在我的心裡世界增添一顆明亮而耀眼的星!
咱們下期再見!