之前项目使用的是nacos作为配置中心,使用过程中,还是有不少问题的:
spring.cloud.nacos.config.shared-configs
配置基于上述原因,在新项目决定弃用nacos,改用SpringCloud原生的配置中心,理由:
application.yml
影响所有环境的所有模块application-test.yml
影响test环境的所有模块xxx.yml
影响所有环境的xxx这个指定模块xxx-test.yml
影响test环境的xxx这个指定模块当然,nacos有个优点,它集成了配置自动刷新能力;
而SpringCloud的配置中心,需要集成kafka或rabbitmq之类的消息中间件,才能有配置自动刷新能力。
下面介绍一下SpringCloud的配置中心使用。
本文基于:
可以在git上新建一个项目存储,
也可以在现有的git项目里,新建一个子目录,专用于存储所有配置
我在gitee的一个现有项目里,新建了一个目录,叫 spring-configs
,并事先上传了几个配置文件:
注:实际测试中,发现github基本都是连不上网络的状态,所以改用gitee来测试
在IDEA里新建项目,配置好名称、语言、组、类型、软件包、JDK版本、java版本等:
Config Server
main
函数所在类,添加注解@EnableConfigServer
import org.springframework.cloud.config.server.EnableConfigServer;
@SpringBootApplication
@EnableConfigServer
public class MyConfigServerDemoApplication {
public static void main(String[] args) {
SpringApplication.run(MyConfigServerDemoApplication.class, args);
}
}
application.properties
,并新建文件application.yml
application.properties
application.yml
,添加git配置:server:
port: 8999 # 配置中心服务端在8999端口监听
spring:
application:
name: my-config-server-demo
# 配置完直接启动即可,访问方式:
# http://localhost:8999/{spring.application.name}/{spring.profiles.active}/{git分支}
# {spring.application.name} 必需,就是具体项目的项目名(yml里配置的),不是config-server的哦
# {spring.profiles.active} 必需,就是具体项目的环境(yml里配置的),注:具体项目可以不配置,使用默认值
# {git分支} 可空,git配置文件所在的分支,默认使用下面的default-label
#
# 注意:server项目启动后,会把git项目clone到本地,如windows系统会在 C:\Users\xxx\AppData\Local\Temp\config-repo-xxx
cloud:
config:
server:
git:
uri: https://gitee.com/youbl/my-demo.git # git项目的url地址,支持gitlab、github、gitee等
search-paths: spring-configs # git项目下,存放yml配置文件的子目录
username: beinet # git账号
password: 123456 # git密码
default-label: master # 默认会获取main分支,不存在就报错: No such label: main
timeout: 6 # 读取git的超时时间,默认5秒
#clone-on-start: true # 启动时把配置clone到本地,默认false,第一次访问会比较慢
#basedir: c:/abc # 本地默认工作目录
#refresh-rate: 100 # 服务从git更新配置的时间间隔,单位秒,默认值0,表示每次请求都去获取最新配置
skip-ssl-validation: true # 忽略git地址的ssl错误: unable to find valid certification path to requested target
OK,启动项目预览下
启动上面的项目,启动完成后,访问地址:http://localhost:8999/config-client-demo/test
可以看到类似如下的json,列出了该项目会访问到的所有yml文件,以及内容:
{
"name": "config-client-demo",
"profiles": [
"test"
],
"label": null,
"version": "169ff9974ccc09c594995ceabac26c983cd607bf",
"state": null,
"propertySources": [
{
"name": "https://gitee.com/youbl/my-demo.git/spring-configs/config-client-demo-test.yml",
"source": {
"beinet.config": "配置中心的front-study-test值",
"beinet.tttt": "abdefssadfga"
}
},
{
"name": "https://gitee.com/youbl/my-demo.git/spring-configs/application-test.yml",
"source": {
"beinet.config": "配置中心的全局-test值",
"beinet.tttt": "abdefs",
"beinet.newVal": "2惹2"
}
},
{
"name": "https://gitee.com/youbl/my-demo.git/spring-configs/config-client-demo.yml",
"source": {
"beinet.config": "配置中心的front-study默认值",
"beinet.tttt": "abdefs"
}
},
{
"name": "https://gitee.com/youbl/my-demo.git/spring-configs/application.yml",
"source": {
"beinet.config": "配置中心的全局值",
"beinet.tttt": "abdefs"
}
}
]
}
配置中心服务端搭建完成,下一步就是配置客户端去连接和访问配置中心的配置了。
Config Client
即可完成依赖添加。 <dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-bootstrapartifactId>
dependency>
<parent>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-parentartifactId>
<version>2.7.9version>
<relativePath/>
parent>
<properties>
<java.version>1.8java.version>
<spring-cloud.version>2021.0.6spring-cloud.version>
properties>
<dependencies>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-configartifactId>
dependency>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-bootstrapartifactId>
dependency>
dependencies>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-dependenciesartifactId>
<version>${spring-cloud.version}version>
<type>pomtype>
<scope>importscope>
dependency>
dependencies>
dependencyManagement>
bootstrap.yml
配置在项目的resources
目录下,新建文件bootstrap.yml
,内容参考:
spring:
application:
name: config-client-demo # 项目名,配置中心读取配置用
cloud:
config:
uri: http://localhost:8999 # 指定配置中心的url
profile: test # 指定使用哪个配置,可以搭配spring.profiles.active使用
label: master # 指定分支,可为空,默认取主干
注意:如果报错:No spring.config.import property has been defined
要确认下Maven依赖是不是少了spring-cloud-starter-bootstrap
OK,启动客户端,输出一下在配置中心写的配置试试吧
测试的参考代码:
package beinet.cn.configclientdemo;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.core.env.Environment;
@SpringBootApplication
public class ConfigClientDemoApplication implements CommandLineRunner {
public static void main(String[] args) {
SpringApplication.run(ConfigClientDemoApplication.class, args);
}
@Autowired
Environment env;
@Override
public void run(String... args) throws Exception {
System.out.println(env.getProperty("beinet.config"));
}
}
客户端成功连接上配置中心后,只能在启动时读取配置,
如果git上的配置进行了修改,必须重启客户端,才能加载到最新的配置。
这里介绍一下,如何手动刷新客户端配置。
打开客户端项目config-client-demo
的pom.xml,添加如下依赖:
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-actuatorartifactId>
dependency>
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
打开客户端项目config-client-demo
的bootstrap.yml
,添加如下配置:
management:
endpoints:
web:
exposure:
include: "refresh" # 用*表示暴露全部
RefreshScope
注解在读取了配置,且需要刷新的类上添加RefreshScope
注解,如:
@Component
@RefreshScope
public class ConfigTest2 {
@Value("${beinet.newVal}")
String str3;
public String getStr3() {
return str3;
}
}
OK,当配置中心的配置发生了变更时,手工调用客户端的接口即可完成配置刷新:
POST http://localhost:8080/actuator/refresh
对应的CURL命令: curl -X POST http://localhost:8080/actuator/refresh
执行完接口请求,可以看到客户端的日志,会多出重新加载配置的日志,例如:
2023-03-22 13:38:37.976 INFO 2260 --- [nio-8080-exec-8] c.c.c.ConfigServicePropertySourceLocator : Fetching config from server at : http://localhost:8999
2023-03-22 13:38:38.844 INFO 2260 --- [nio-8080-exec-8] c.c.c.ConfigServicePropertySourceLocator : Located environment: name=config-client-demo, profiles=[test], label=master, version=e5b67cb61ad976c2c1bd863dae769aabf1553c21, state=null
2023-03-22 13:38:38.844 INFO 2260 --- [nio-8080-exec-8] b.c.PropertySourceBootstrapConfiguration : Located property source: [BootstrapPropertySource {name='bootstrapProperties-configClient'}, BootstrapPropertySource {name='bootstrapProperties-https://gitee.com/youbl/my-demo.git/spring-configs/config-client-demo-test.yml'}, BootstrapPropertySource {name='bootstrapProperties-https://gitee.com/youbl/my-demo.git/spring-configs/application-test.yml'}, BootstrapPropertySource {name='bootstrapProperties-https://gitee.com/youbl/my-demo.git/spring-configs/config-client-demo.yml'}, BootstrapPropertySource {name='bootstrapProperties-https://gitee.com/youbl/my-demo.git/spring-configs/application.yml'}]
2023-03-22 13:38:38.846 INFO 2260 --- [nio-8080-exec-8] o.s.boot.SpringApplication : No active profile set, falling back to 1 default profile: "default"
2023-03-22 13:38:38.849 INFO 2260 --- [nio-8080-exec-8] o.s.boot.SpringApplication : Started application in 0.998 seconds (JVM running for 402.306)
配置中心的front-study-test值..
Environment
的Bean读取配置,会被刷新RefreshScope
注解的配置,不会被刷新RefreshScope
也不行上面的方法,只能刷新单个客户端的配置,不会刷新其它服务的配置,
甚至,如果某服务部署了多台机器,那么其它机器也不会被刷新。
你需要知道每个服务部署了哪些机器,一台机器一台机器的去调用接口刷新。
这一节,介绍一下如何在配置中心服务端通知所有客户端去自动刷新配置。
原理说明:
pom.xml
,添加如下依赖:<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-actuatorartifactId>
dependency>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-bus-kafkaartifactId>
dependency>
application.yml
,添加如下配置,启用bus消息推送:spring:
cloud:
# 启用bus后,对应的消息队列(如kafka)会自动创建一个topic springCloudBus
# 使用 curl -X POST http://localhost:8999/actuator/busrefresh 会触发消息推送到这个topic,供client使用
bus:
refresh:
enabled: true
kafka:
bootstrap-servers: 10.1.2.3:9092 # 用到的Kafka连接信息
management:
endpoints:
web:
exposure:
include: "busrefresh" # 用*表示暴露全部
OK,可以启动配置中心服务端了。
pom.xml
,添加如下依赖:<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-actuatorartifactId>
dependency>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-bus-kafkaartifactId>
dependency>
bootstrap.yml
,添加kafka配置,启用bus消息消费:spring:
kafka:
bootstrap-servers: 10.1.2.3:9092 # 用到的Kafka连接信息
OK,把这些客户端启动吧。
OK,当配置中心的配置发生了变更时,
还是要手工调用接口,但是是调用服务端的接口,即可完成所有客户端的配置刷新:
POST http://localhost:8999/actuator/busrefresh
对应的CURL命令: curl -X POST http://localhost:8999/actuator/busrefresh
执行完接口请求,可以看到服务端,以及每个客户端,都会多出重新加载配置的日志
最后,原生的配置中心就是这点不好,改了配置,一定要手工调接口,才能刷新。