SpringCloud 是微服务中的翘楚,最佳的落地方案。
Spring Cloud Config 是一个解决分布式系统的配置管理方案,它包含了 server 和 client 两个部分。
server 用来获取远程的配置信息(默认为 Git 仓库),并且以接口的形式提供出去;
client 根据 server 提供的接口读取配置文件,以便于初始化自己的应用。
在这里存在一个问题,当配置中心(远程 Git 仓库等)中的配置发生了变化,server 是可以获取到最新的配置,
但是 client 不可以(在下面有验证);
这时,就用到了 SpringCloud 另一个组件:Spring Cloud Bus。
Spring Cloud Bus 通过轻量消息代理连接各个分布的节点。简单的说,就是把多个应用之间通过 AMQP 消息
代理建立联系。
Spring Cloud Bus 在项目中的作用之一就是利用 MQ 的广播机制在分布式系统中传播消息,目前常用的有 Kafka
和 RabbitMQ。本博客中使用 Kafka 作为消息中间件。
GitHub地址:https://github.com/intomylife/SpringCloud
4.0.0
com.zwc
springcloud-config-bus-commons
1.0
springcloud-config-bus-commons
公用工程
jar
UTF-8
1.8
Cairo-SR3
Finchley.RELEASE
org.springframework.cloud
spring-cloud-starter-bus-kafka
io.spring.platform
platform-bom
${platform-bom.version}
pom
import
org.springframework.cloud
spring-cloud-dependencies
${spring-cloud-dependencies.version}
pom
import
org.springframework.boot
spring-boot-maven-plugin
# kafka 配置
## kafka 服务地址
spring.kafka.bootstrap-servers=127.0.0.1:9092
## producer 提供者
### 如果该值大于零时,表示启用重试失败的发送次数
spring.kafka.producer.retries=0
### 每次批量发送消息的数量
spring.kafka.producer.batch-size=16384
spring.kafka.producer.buffer-memory=33554432
### 指定消息 key 和消息体的编解码方式
spring.kafka.producer.key-serializer=org.apache.kafka.common.serialization.StringSerializer
spring.kafka.producer.value-serializer=org.apache.kafka.common.serialization.StringSerializer
## consumer 消费者
### 指定默认消费者 group id
spring.kafka.consumer.group-id=springcloud-config-bus-group
### 当 Kafka 中没有初始偏移量或者服务器上不再存在当前偏移量时该怎么办,默认值为 latest,表示自动将偏移重置为最新的偏移量,可选的值为 latest, earliest, none
spring.kafka.consumer.auto-offset-reset=earliest
### 如果为 true,则消费者的偏移量将在后台定期提交,默认值为 true
spring.kafka.consumer.enable-auto-commit=false
### 如果 'enable.auto.commit'为true,则消费者偏移自动提交给 Kafka 的频率(以毫秒为单位),默认值为 5000
spring.kafka.consumer.auto-commit-interval=100
### 指定消息 key 和消息体的编解码方式
spring.kafka.consumer.key-deserializer=org.apache.kafka.common.serialization.StringDeserializer
spring.kafka.consumer.value-deserializer=org.apache.kafka.common.serialization.StringDeserializer
package com.zwc.core.config;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
/*
* @ClassName InitConfig
* @Desc TODO 加载配置文件
* @Date 2019/6/15 12:58
* @Version 1.0
*/
@Configuration
@PropertySource("classpath:system.properties")
public class InitConfig {
}
创建一个 Git 库,里面存放配置文件,文件夹名称为:config-repo;这里模拟不同的环境,所以分别构建了
dev(开发)、uat(测试)以及 online(线上)三种不同的配置文件;此文件夹存在此项目的根目录中
- springcloud-config-bus
- config-repo
- system-dev.properties
- system-online.properties
- system-uat.properties
+ springcloud-config-bus-commons
+ springcloud-config-bus-service
下面的 server 和 client 项目启动前都需要启动 zkServer 和 kafka
zkServer 的使用很简单,下载 zip 后解压双击打开就可以用了
kafka 的使用可以参考 SpringBoot整合Kafka 博客末尾扩展部分,有写 windows 本地安装和启动 kafka
① 此工程下有四个模块:一个注册中心,一个 server 以及二个 client
② 二个 client 除端口、应用名称以及 group-id 不一致以外,其他代码基本一致
4.0.0
com.zwc
springcloud-config-bus-service
1.0
com.zwc
springcloud-config-bus-registry-service
1.0
springcloud-config-bus-registry-service
注册中心
jar
org.springframework.cloud
spring-cloud-starter-netflix-eureka-server
org.springframework.boot
spring-boot-maven-plugin
# 端口
server:
port: 8761
# 应用名称
spring:
application:
name: eureka-server
eureka:
instance:
# 使用 ip 代替实例名
prefer-ip-address: true
# 实例的主机名
hostname: ${spring.cloud.client.ip-address}
# 实例的 ID 规则
instance-id: ${spring.cloud.client.ip-address}:${spring.application.name}:${server.port}
client:
# 是否向注册中心注册自己
registerWithEureka: false
# 是否向注册中心获取注册信息
fetchRegistry: false
serviceUrl:
# 注册中心地址
defaultZone: http://${eureka.instance.hostname}:${server.port}/eureka/
package com.zwc;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.server.EnableEurekaServer;
@SpringBootApplication
@EnableEurekaServer
public class SpringcloudConfigBusRegistryServiceApplication {
public static void main(String[] args) {
SpringApplication.run(SpringcloudConfigBusRegistryServiceApplication.class, args);
}
}
1. 项目启动成功后访问 http://localhost:8761/ 即可看到 eureka-server 主页面
4.0.0
com.zwc
springcloud-config-bus-service
1.0
com.zwc
springcloud-config-bus-server-service
1.0
springcloud-config-bus-server-service
config server
jar
com.zwc
springcloud-config-bus-commons
1.0
org.springframework.cloud
spring-cloud-starter-netflix-eureka-client
org.springframework.cloud
spring-cloud-config-server
org.springframework.boot
spring-boot-starter-actuator
org.springframework.boot
spring-boot-maven-plugin
# 端口
server:
port: 8000
spring:
application:
# 应用名称
name: config-bus-server
cloud:
config:
server:
git:
# 仓库地址
uri: https://github.com/intomylife/SpringCloud
# 对应 {label} 部分,即 Git 的分支
label: master
# 仓库文件夹名称,多个以逗号分隔
search-paths: springcloud-config-bus/config-repo
# git 仓库用户名(公开库可以不用填写)
username:
# git 仓库密码(公开库可以不用填写)
password:
bus:
# 开启消息跟踪
enabled: true
trace:
enabled: true
kafka:
consumer:
# 指定默认消费者 group id
# 如果不设置,将会使用 commons 工程中的 group-id,那时调用 /actuator/bus-refresh 接口只会刷新其中一个 client
# 因为在同一个组中的 Consumer,同一个主题只会被一个 Consumer 接收
group-id: config-bus-server-group
eureka:
instance:
# 使用 ip 代替实例名
prefer-ip-address: true
# 实例的主机名
hostname: ${spring.cloud.client.ip-address}
# 实例的 ID 规则
instance-id: ${spring.cloud.client.ip-address}:${spring.application.name}:${server.port}
client:
serviceUrl:
# 注册中心地址
defaultZone: http://${eureka.instance.hostname}:8761/eureka/
management:
endpoints:
web:
exposure:
# 开启刷新端点
include: bus-refresh
package com.zwc;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.config.server.EnableConfigServer;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
@SpringBootApplication
@EnableConfigServer
@EnableEurekaClient
public class SpringcloudConfigBusServerServiceApplication {
public static void main(String[] args) {
SpringApplication.run(SpringcloudConfigBusServerServiceApplication.class, args);
}
}
1. 项目启动成功后访问:http://localhost:8761/(注册中心)可以看到服务已经被注册进来了
2. 访问地址:http://localhost:8000/config-repo/dev(获取完整配置信息)
3. 输出内容:
{
"name":"config-repo",
"profiles":[
"dev"
],
"label":null,
"version":"742276b4c0d821af224cb104590a5d93d4e5e71a",
"state":null,
"propertySources":[
]
}
4. 访问地址:http://localhost:8000/system/dev(获取完整配置信息)
5. 输出内容:
{
"name":"system",
"profiles":[
"dev"
],
"label":null,
"version":"742276b4c0d821af224cb104590a5d93d4e5e71a",
"state":null,
"propertySources":[
{
"name":"https://github.com/intomylife/SpringCloud/springcloud-config-bus/config-repo/system-dev.properties",
"source":{
"hello":"I'm dev."
}
}
]
}
6. 访问地址:http://localhost:8000/system-dev.properties(获取指定配置文件内容)
7. 输出内容:'hello: I'm dev.'
8. 证明 server 已经成功从远程仓库中获取到了配置信息
注:接口访问有如下规则:
/{application}/{profile}[/{label}]
/{application}-{profile}.yml
/{label}/{application}-{profile}.yml
/{application}-{profile}.properties
/{label}/{application}-{profile}.properties
注:一共有两个 client,除端口、应用名称以及 group-id 以外的代码基本上一致,所有这里就列举其中一个
4.0.0
com.zwc
springcloud-config-bus-service
1.0
com.zwc
springcloud-config-bus-clientfirst-service
1.0
springcloud-config-bus-clientfirst-service
config client
jar
com.zwc
springcloud-config-bus-commons
1.0
org.springframework.cloud
spring-cloud-starter-config
org.springframework.cloud
spring-cloud-starter-netflix-eureka-client
org.springframework.boot
spring-boot-starter-actuator
org.springframework.boot
spring-boot-maven-plugin
# 端口
server:
port: 9000
spring:
application:
# 应用名称
name: config-bus-client-first
spring:
cloud:
config:
# 对应 {label} 部分,即 Git 的分支
label: master
# 对应 {application} 部分
name: system
# 对应 {profile} 部分
profile: dev
discovery:
# 开启 Config 服务发现与注册
enabled: true
# 指定 server
service-id: config-bus-server
bus:
# 开启消息跟踪
enabled: true
trace:
enabled: true
kafka:
consumer:
# 指定默认消费者 group id
# 如果不设置,将会使用 commons 工程中的 group-id,那时调用 /actuator/bus-refresh 接口只会刷新其中一个 client
# 因为在同一个组中的 Consumer,同一个主题只会被一个 Consumer 接收
group-id: config-bus-client-first-group
eureka:
instance:
# 使用 ip 代替实例名
prefer-ip-address: true
# 实例的主机名
hostname: ${spring.cloud.client.ip-address}
# 实例的 ID 规则
instance-id: ${spring.cloud.client.ip-address}:${spring.application.name}:${server.port}
client:
serviceUrl:
# 注册中心地址
defaultZone: http://${eureka.instance.hostname}:8761/eureka/
management:
endpoints:
web:
exposure:
# 开启刷新端点
include: bus-refresh
package com.zwc.first.controller;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cloud.context.config.annotation.RefreshScope;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* @ClassName HelloController
* @Desc TODO 读取 server 配置文件
* @Date 2019/6/2 16:54
* @Version 1.0
*/
@RestController
@RefreshScope
public class HelloController {
@Value("${hello}")
private String hello;
/*
* @ClassName HelloController
* @Desc TODO 读取 server 配置文件
* @Date 2019/6/2 16:56
* @Version 1.0
*/
@RequestMapping("/hello")
public String hello() {
return this.hello;
}
}
package com.zwc;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
@SpringBootApplication
@EnableEurekaClient
public class SpringcloudConfigBusClientfirstServiceApplication {
public static void main(String[] args) {
SpringApplication.run(SpringcloudConfigBusClientfirstServiceApplication.class, args);
}
}
1. 项目启动成功后访问:http://localhost:8761/(注册中心)可以看到服务已经被注册进来了
2. 访问地址:http://localhost:9000/hello
3. 输出内容:'I'm dev.'
4. 启动另一个 client(springcloud-config-bus-clientsecond-service)
5. 项目启动成功后访问:http://localhost:8761/(注册中心)可以看到服务已经被注册进来了
6. 访问地址:http://localhost:9001/hello
7. 输出内容:'I'm dev.'
8. 证明两个 client 都已经成功从注册中心的 server 中读取到了配置信息
9. 全局刷新
10. 修改远程 Git 仓库 system-dev.properties 配置文件中的内容为 'hello=I'm dev. update!!!'
11. 此时刷新地址:http://localhost:8000/system/dev(获取完整配置信息)
12. 输出内容:
{
"name":"system",
"profiles":[
"dev"
],
"label":null,
"version":"4c89d3baf3a67e56f5afe3f97914a41c16f46e7d",
"state":null,
"propertySources":[
{
"name":"https://github.com/intomylife/SpringCloud/springcloud-config-bus/config-repo/system-dev.properties",
"source":{
"hello":"I'm dev. update!!!"
}
}
]
}
13. 与前言中说的一致,server 可以及时获取到最新的配置信息
14. 这时再刷新地址:http://localhost:9000/hello 和 http://localhost:9001/hello
15. 发现依旧还是: 'I'm dev.'
16. 打开 cmd,输入命令: curl -X POST http://localhost:8000/actuator/bus-refresh
17. 此时 cmd 中正常情况下是不会有任何输出内容的
18. 再次刷新地址:http://localhost:9000/hello 和 http://localhost:9001/hello
19. 发现内容都更改为:'I'm dev. update!!!'
20. 局部刷新
21. 再次修改远程 Git 仓库 system-dev.properties 配置文件中的内容为 'hello=I'm dev. update 222!!!'
22. 此时刷新地址:http://localhost:8000/system/dev(获取完整配置信息)
23. 输出内容:
{
"name":"system",
"profiles":[
"dev"
],
"label":null,
"version":"96e58a513e38c96e22361e0bf97a1735f448bf59",
"state":null,
"propertySources":[
{
"name":"https://github.com/intomylife/SpringCloud/springcloud-config-bus/config-repo/system-dev.properties",
"source":{
"hello":"I'm dev. update 222!!!"
}
}
]
}
24. server 依旧及时获取到最新的配置信息
25. 这时再刷新地址:http://localhost:9000/hello 和 http://localhost:9001/hello
26. 发现依旧还是: 'I'm dev. update!!!'
27. 打开 cmd,输入命令: curl -X POST http://localhost:8000/actuator/bus-refresh/config-bus-client-first:9000
28. 此时 cmd 中正常情况下是不会有任何输出内容的
29. 再次刷新地址:http://localhost:9000/hello
30. 发现内容更改为:'I'm dev. update 222!!!'
31. 再次刷新地址:http://localhost:9001/hello
32. 发现内容还是:'I'm dev. update!!!'
33. 所以全局刷新就使用 /actuator/bus-refresh 接口,局部刷新就使用 /actuator/bus-refresh/应用名称:端口
希望能够帮助到你
over