Spring Cloud 服务注册与发现源码笔记 (Nacos/Consul/Eureka)

三合一唉!!!! 超级尊享版!!!!! 肝了一天唉!!!!

Eureka

关键类

# 服务注册
1.EurekaClientAutoConfiguration
    注册了众多的 bean
    一部分用于和 Eureka Server 交互
    一部分和 Commons 项目对接
    注册了(EurekaClient/EurekaAutoServiceRegistration/ApplicationInfoManager/EurekaRegistration)
       
2.EurekaClient
    与 Eureka Server 端交互
    负责向 Eureka Server 端注册/注销服务实例
    在构造方法和 shutdown 方法中根据配置处理自动注册和自动注销.

3.InstanceInfoReplicator
    负责定义一个注册或更新服务实例的任务
    负责管理任务执行器

4.RestTemplateEurekaHttpClient
    负责根据服务实例信息构造注册/注销的 Http 请求
    使用 RestTemplate 发送请求
    
5.EurekaAutoServiceRegistration
    负责管理服务实例的自动注册/注销, 与容器生命周期挂钩

6.ApplicationInfoManager
    负责管理服务实例状态变更事件(即管理监听者并在适当时机触发他们)

7.ApplicationInfoManager.StatusChangeListener
    状态改变事件监听者类
    
8.EurekaHealthCheckHandler
    服务实例状态健康检查类, 注册/注销服务实例前会调用此类进行检查其状态


# 服务发现
1.BlockingLoadBalancer (Commons 项目上)
    负责调用实际的负载均衡器去选择一个服务实例
    负责调度负载均衡整个过程, 触发相应的生命周期.
    
2.RoundRobinLoadBalancer (Commons 项目上)
    实际的负载均衡策略算法类
    负责连接 ServiceInstanceListSupplier 从其中获取服务实例列表

3.ServiceInstanceListSupplier
    定义了获取服务实例实例列表的接口(任意方式)

4.DiscoveryClient
    定义了从服务端获取服务实例列表的接口(更明确了)

4.DiscoveryClientServiceInstanceListSupplier
    ServiceInstanceListSupplier 的实现类
    连接 ReactiveDiscoveryClient 类, 将服务端获取道德服务实例列表返回出去
    
5.EurekaDiscoveryClient
    实现了 DiscoveryClient 接口
    负责调用 EurekaClient 从 Eureka Server 端获取服务实例列表.

6.EurekaClient
    与 Eureka Server 端交互
    负责向 Eureka Server 端获取服务实例列表
    在构造方法和 shutdown 方法中根据配置处理自动注册和自动注销.

7.EurekaServiceInstance
    服务实例信息对象
    实现 ServiceInstance, 与 Commons 对接

# 总结
实现了 Commons 的 DiscoveryClient 接口(即 EurekaDiscoveryClient), 于是服务发现实现了;
实现了 Commons 的 ServiceRegisty 接口(即 EurekaServiceRegistry), 于是服务注册也实现了.
再总结: Commons 大发好.

服务注册流程

# EurekaClient 构造方法触发(前提 shouldRegisterWithEureka 为 true)
1.EurekaClientAutoConfiguration 让容器里注册了一个 EurekaClient
2.EurekaClient 这个类在构造方法中的 initScheduledTasks() 生成了一个 InstanceInfoReplicator 对象
3.然后调用其 instanceInfoReplicator.start(), 逻辑是添加一个定时任务(仅执行一次), 定时任务执行 run()
4.run() 里面会执行 discoveryClient.register() 也就是注册实例信息到 Eureka 上去.
5.register() 里面会调用 RestTemplateEurekaHttpClient#register() 
6.这个方法是构造一个 HTTP 请求, 地址为 serviceUrl + "apps/" + info.getAppName(), Method 为 POST, 即 Eureka server 的ip/apps/服务实例名称(如spring.application.name), 当然请求 body 还会带上实例信息 info 对象.

# 根据容器生命周期触发
1.EurekaClientAutoConfiguration 让容器里注册了一个 EurekaAutoServiceRegistration.
2.这个类即是 SmartLifecycle(与容器生命周期绑定), 也是 SmartApplicationListener(监听容器加载/关闭事件)
3.因此其对应的 start()/stop() 和 onApplicationEvent() 都实现了对应的逻辑.
4.如 start() 的逻辑为调用 EurekaServiceRegistry#register()
5.register() 先修改本地服务实例的装填, 再通过 com.netflix.discovery.DiscoveryClient#registerHealthCheck() 来往任务管理器中提交一个任务, 后台执行, 任务 InstanceInfoReplicator#onDemandUpdate()
6.此任务就是最终也会调用前面提到的 (4) 中的 run(). 然后就注册上去了.

# PS
(6) 中的任务, 与状态监听是一致的

总结: 容器注册 EurekaClient, 调用构造方法完成大量初始化工作后, 另起一个线程调用 restTemplate 发送 HTTP 请求将当前服务实例信息发送给 Eureka server. 完成服务注册工作.

graph TB

A(EurekaClientAutoConfiguration)
A1(EurekaClient)
A2(InstanceInfoReplicator)
A3(EurekaClient#register)
A4(RestTemplateEurekaHttpClient)
A5(RestTemplate)
A6(Eureka Server)
A7(EurekaHttpResponse)

A--让容器里注册一个-->A1
A1--在构造方法中生成一个-->A2
A2--在 run 方法中调用-->A3
A3--又把服务实例信息交给-->A4
A4--根据服务实例信息构造HTTP请求-->A5
A5--将服务实例信息发送给-->A6
A6--返回一个-->A7


image-20210129171345329

服务注销流程

# 容器关闭(配置了容器 shouldUnregisterOnShutdown=true)
1.在 DiscoveryClient#shutdown() 中根据 shouldUnregisterOnShutdown 判断是否需要注销
2.然后在 unregister() 中调用 RestTemplateEurekaHttpClient#cancel() 
3.cancel() 中使用 restTemplate 构造 Method 为 DELETE, URL 为 serviceUrl + "apps/" + appName + '/' + id 的请求并发送给 Eureka server 告知其注销 appName 下对应的服务实例(根据 id).

# 接收到容器关闭事件
1.EurekaClientAutoConfiguration 让容器里注册了一个 EurekaAutoServiceRegistration.
2.这个类即是 SmartLifecycle(与容器生命周期绑定), 也是 SmartApplicationListener(监听容器加载/关闭事件)
3.因此其对应的 start()/stop() 和 onApplicationEvent() 都实现了对应的逻辑.
4.如 stop() 的逻辑为调用 EurekaServiceRegistry#deregister()
5.deregister() 的作用是执行 ApplicationInfoManager#setInstanceStatus() 将状态改为 DOWN
6.因为 ApplicationInfoManager 这个类专门管理状态变化事件, 因此还会将事件发布出去. 
7.而在 initScheduledTasks 中就添加了这样的一个监听者, 起作用为调用 InstanceInfoReplicator#onDemandUpdate()
8.接着会调用 InstanceInfoReplicator#run(), 第一行代码 refreshInstanceInfo() 会确保状态为最新的(那也还是 DOWN)
9.但是其最终并不是调用 cancel 注销, 而 register 注册, 不过其中的状态是 DOWN, 因此会有 server 那边判断; 所以容器关闭事件并不会触发 cancel, 但效果应是一样的.

总结: 要么是 shutdown() 中 unregister 调用了 cancel(), 发出了 Method 为 DELETE 的请求来注销; 要么就是 EurekaClientAutoConfiguration 中与容器生命周期做关联, 全程使用 register 接口来更新状态.

graph TD

A(容器关闭#close)
A1(DiscoveryClient#shutdown)
A2(DiscoveryClient#unregister)
A3(RestTemplateEurekaHttpClient#cancel)
A4(RestTemplate)
A5(Eureka Server) 

A--触发-->A1
A1--调用-->A2
A2--调用-->A3
A3--构造HTTP请求交给-->A4
A4--发送注销申请给-->A5

image-20210129171237603

服务发现流程

1.Spring Cloud Commons 中注册了一个 ServiceInstanceListSupplier, 具体为(DiscoveryClientServiceInstanceListSupplier)
2.这个类的作用是借助 ReactiveDiscoveryClient 的 getInstances(String serviceId) 方法向 LoadBalancer 提供从具体的 server(如Eureka) 获取服务实例对象列表, 这样只要实现 ReactiveDiscoveryClient 并放入容器就可以和 Spring cloud LoadBalancer 对接了.
3.在 EurekaDiscoveryClientConfiguration 中让容器注册了一个 DiscoveryClient(具体为 EurekaDiscoveryClient).
4.因为 Spring Cloud Commons 做了大量的预备对接工作, 所以对接其实就结束了.
5.那再简单说下 EurekaDiscoveryClient 的实现, 即注入一个 EurekaClient eurekaClient, 然后调用 DiscoveryClient#getInstancesByVipAddress() 就获取到了 ServiceInstance 列表.

总结: Commons 中准备好了对接的方式: 实现 DiscoveryClient 接口, 接着我们的确实现了 DiscoveryClient 接口, 即 EurekaDiscoveryClient, 而这这个类则会调用 EurekaClient 的 getInstancesByVipAddress 从 Eureka Server 端获取注册了的服务实例信息.

graph TD

A1(BlockingLoadBalancerClient#位于Commons项目)
A2(RoundRobinLoadBalancer#位于Commons项目)
A4(LoadBalancerClientConfiguration#位于Commons项目)
A5(DiscoveryClientServiceInstanceListSupplier)
A6(EurekaDiscoveryClient#getInstances)
A7(EurekaClient#getInstancesByVipAddress)
A9(Eureka Server)
A8(ServiceInstance 集合)

A1--choose 方法调用-->A2
A2--choose 方法调用-->A5
A4--注入一个-->A5
A5--调用-->A6
A6--调用-->A7
A7--从-->A9
A9--获取-->A8
A10(EurekaDiscoveryClientConfiguration)--注入一个-->A6

image-20210129233854498

Consul

关键类

# 服务注册
1.ConsulAutoServiceRegistrationAutoConfiguration
    负责添加自动注册/注销相关的 bean
    注册了 ConsulAutoServiceRegistration/ConsulAutoServiceRegistrationListener/ConsulAutoRegistration
    
2.ConsulServiceRegistryAutoConfiguration
    负责注册服务注册相关的 bean
    注册了 ConsulServiceRegistry
    
3.ConsulDiscoveryClientConfiguration
    负责注册服务发现相关的 bean
    注册了 ConsulDiscoveryClient
    
4.ConsulServiceRegistry
    负责与 ConsulClient 对接, 再提供注册/注销功能

5.ConsulClient
    与 Consul Server 端交互
    负责向 Consul Server 端注册/注销服务实例

6.AgentConsulClient
    负责根据服务实例信息构造注册/注销的 Http 请求
    
7.ConsulAutoServiceRegistration/ConsulAutoServiceRegistrationListener
    负责管理服务实例的自动注册/注销, 与容器生命周期挂钩



# 服务发现
1.BlockingLoadBalancer (Commons 项目上)
    负责调用实际的负载均衡器去选择一个服务实例
    负责调度负载均衡整个过程, 触发相应的生命周期.
    
2.RoundRobinLoadBalancer (Commons 项目上)
    实际的负载均衡策略算法类
    负责连接 ServiceInstanceListSupplier 从其中获取服务实例列表

3.ServiceInstanceListSupplier
    定义了获取服务实例实例列表的接口(任意方式)

4.DiscoveryClient
    定义了从服务端获取服务实例列表的接口(更明确了)

4.DiscoveryClientServiceInstanceListSupplier
    ServiceInstanceListSupplier 的实现类
    连接 ReactiveDiscoveryClient 类, 将服务端获取道德服务实例列表返回出去
    
5.ConsulDiscoveryClient
    实现了 DiscoveryClient 接口
    负责调用 ConsulClient 从 Consul Server 端获取服务实例列表.

6.ConsulClient
    与 Consul Server 端交互
    负责向 Consul Server 端获取服务实例列表

7.ConsulServiceInstance
    服务实例信息对象
    实现 ServiceInstance, 与 Commons 对接

# 总结
实现了 Commons 的 DiscoveryClient 接口(即 ConsulDiscoveryClient), 于是服务发现实现了;
实现了 Commons 的 ServiceRegisty 接口(即 ConsulServiceRegistry), 于是服务注册也实现了.
再总结: Commons 大发好.

服务注册流程

1.ConsulAutoServiceRegistration 调用 org.springframework.cloud.consul.serviceregistry.ConsulServiceRegistry#register() 完成注册
2.接着 register() 调用 ConsulClient#agentServiceRegister()
3.然后会调用 AgentConsulClient#agentServiceRegister()
4.生成并发送请求, 请求地址为 /v1/agent/service/register

总结: 其实和 Eureka 差不多, 只是我跳过了一点点细节. 基本是就是 与 ServiceRegistry 交互了, 非常的配合 Commons 项目, 就像一个人写的一样...

服务发现流程

1.首先是 DiscoveryClientServiceInstanceListSupplier 会调用 ConsulDiscoveryClient#getInstances()
2.接着 getInstances() 会调用 ConsulClient#getHealthServices()
3.然后就是发送 HTTP 请求了, 请求地址为 /v1/health/service/, 返回的对象封装处理下得到 ServiceInstance 的实现类 ConsulServiceInstance.

总结: 这次真的是和 Eureka 类似, 从 Commons 到 ConsulDiscoveryClient, 流程是一样的. 而其实 ConsulDiscoveryClient 的逻辑也和 EurekaDiscoverClient 类似... 只能说其实服务注册发现这个框架, 我们关注的功能其实并不是难点. 难点是 server 端的管理.

Nacos

关键类

# 服务注册
1.NacosServiceRegistryAutoConfiguration
    负责添加自动注册/注销相关的 bean
    注册了 NacosAutoServiceRegistration/NacosServiceRegistry/NacosRegistration
    
2.NacosDiscoveryClientConfiguration
    负责注册服务发现相关的 bean
    注册了 NacosDiscoveryClient
    
3.NacosServiceRegistry
    负责与 NamingService 对接, 再提供注册/注销功能

4.NamingService
    与 Nacos Server 端交互
    负责向 Nacos Server 端注册/注销服务实例
    调用 NamingProxy

5.NamingProxy
    负责根据服务实例信息构造注册/注销的 Http 请求
    
6.NacosAutoServiceRegistration
    负责管理服务实例的自动注册/注销, 与容器生命周期挂钩

7.NacosRegistration
    本地实例对象, 相比服务实例数据更多


# 服务发现
1.BlockingLoadBalancer (Commons 项目上)
    负责调用实际的负载均衡器去选择一个服务实例
    负责调度负载均衡整个过程, 触发相应的生命周期.
    
2.RoundRobinLoadBalancer (Commons 项目上)
    实际的负载均衡策略算法类
    负责连接 ServiceInstanceListSupplier 从其中获取服务实例列表

3.ServiceInstanceListSupplier
    定义了获取服务实例实例列表的接口(任意方式)

4.DiscoveryClient
    定义了从服务端获取服务实例列表的接口(更明确了)

4.DiscoveryClientServiceInstanceListSupplier
    ServiceInstanceListSupplier 的实现类
    连接 ReactiveDiscoveryClient 类, 将服务端获取道德服务实例列表返回出去
    
5.NacosDiscoveryClient
    实现了 DiscoveryClient 接口
    负责调用 NamingService 从 Nacos Server 端获取服务实例列表.

6.NamingService
    与 Nacos Server 端交互
    负责向 Nacos Server 端获取服务实例列表

7.NacosServiceInstance
    服务实例信息对象
    实现 ServiceInstance, 与 Commons 对接

# 总结
实现了 Commons 的 DiscoveryClient 接口(即 NacosDiscoveryClient), 于是服务发现实现了;
实现了 Commons 的 ServiceRegisty 接口(即 ConsulServiceRegistry), 于是服务注册也实现了.
再总结: Commons 大发好.

服务注册流程

1.NacosAutoServiceRegistration 调用 NacosServiceRegistry#register 完成注册
2.接着 register() 调用 NacosNamingService#registerInstance()
3.然后会调用 NamingProxy#registerService()
4.生成并发送请求, 请求地址为 /nacos/v1/ns/instance

总结: 这和 Consul 差不多; 还是继承 AbstractAutoServiceRegistration 来与 ServiceRegistry 交互了, 也是非常的配合 Commons 项目啊!!

服务发现流程

1.首先是 DiscoveryClientServiceInstanceListSupplier 会调用 NacosDiscoveryClient#getInstances()
2.接着 getInstances() 会调用 NacosServiceDiscovery#getInstances()
3.然后就是发送 HTTP 请求了, 请求地址为 /nacos/v1/ns/instance/list, 返回的对象封装处理下得到 ServiceInstance 的实现类 NacosServiceInstance.

总结: 依然是和 Eureka/Consul 类似, 从 Commons 到 NacosDiscoveryClient, 流程是一样的. 而其实 NacosDiscoveryClient 的逻辑也和 ConsulDiscoveryClient/EurekaDiscoverClient 类似, 最终总是发起 HTTP 查询 server 端, 所以 server 端的代码才比较有趣啊.

Nacos Config

Nacos Config Client 加载 nacos server 配置原理

关键类

1.NacosConfigBootstrapConfiguration
    注册了一个 NacosPropertySourceLocator

2.NacosPropertySourceLocator
    用于从 nacos server 上加载 dataId 对应的配置文件到 environment 的 PropertySource 集合中.
    
3.ConfigService
    用于与 nacos server 通信, 获取配置文件, 乃至订阅更新
    
4.NacosPropertySourceBuilder
    借助 ConfigService 与 NacosDataParserHandler 从 nacos 获取 NacosPropertySource
    
5.NacosDataParserHandler
    用于序列化(转码)服务端的配置文件数据为一个 PropertySource(即 NacosPropertySource)


# 自动刷新
1.NacosConfigAutoConfiguration
2.NacosContextRefresher
    

总结: ConfigService 与 NacosDataParserHandler 是两个干事的, 其他都是渣渣!!!

从 server 获取配置流程

1.在 NacosConfigBootstrapConfiguration 中注册了一个 PropertySourceLocator(即 NacosPropertySourceLocator), 可用于为 environment 添加 PropertySource
2.然后实现 locate 方法, 即 NacosPropertySourceLocator#locate()
3.在 locate() 的最后面, 调用了 loadApplicationConfiguration()
4.这个方法通过多次调用 loadNacosDataIfPresent() 加载 dataId 和不同文件后缀以及profile 组和得到不同的 dataId.
5.loadNacosDataIfPresent() 调用了 loadNacosPropertySource()
6.其最终调用了 NacosPropertySourceBuilder#loadNacosData()
7.loadNacosData() 调用 ConfigService#getConfig() 
8.getConfig() 会使用 ClientWorker#getServerConfig() 发起 HTTP 请求从 nacos server 获取配置文件. 其请求地址是 /v1/cs/configs

还是挺简单的, 发个 HTTP 获取配置文件, 然后转化成一个 PropertySource, 再在 NacosPropertySourceLocator 的 locate 中扔进 environment 中去. Bingo!!!

从 server 实时更新原理

1.在 NacosConfigAutoConfiguration 注册了一个 NacosContextRefresher
2.其实现了 ApplicationListener<ApplicationReadyEvent>, 也就是会在容器准备事件触发后调用 NacosContextRefresher#registerNacosListenersForApplications()
3.在进行两个配置判断后(即 isRefreshEnabled() 和 PropertySource 的 isRefreshable), 调用 NacosContextRefresher#registerNacosListener() 
4.这个方法通过 NacosConfigService#addListener() 注册一个 AbstractSharedListener 监听者到 cacheData.
5.而 ClientWorker#checkConfigInfo() 中添加的 LongPollingRunnable 任务, 会在 run() 中执行 getServerConfig() 获取服务配置文件, 然后与当前缓存对比 md5, 若更新则通过 cacheData 取出刚刚添加的 listener, 触发事件(即 receiveConfigInfo() )
6.触发后又会再发布一个 RefreshEvent 事件
7.接着流转到 RefreshEventListener.onApplicationEvent()
8.然后又会调用 ContextRefresher.refresh(), 这个方法中的 refreshEnvironment() 会调用 addConfigFilesToEnvironment(), 这方法先复制一个 StandardEnvironment, 将其放入到 SpringApplication 中, 再调用 SpringApplication 的 run() 走一遍, 会触发 NacosPropertySourceLocator.locate(), 于是复制的 environment 里面有新数据了, 再将其拷贝替换到当前的 environment. 完成 environment 的刷新. 
9.接着发布一个 EnvironmentChangeEvent 事件, 用来刷新 @ConfigurationProperties 注解的 Bean. (这么喜欢事件?不愧是写出 RocketMQ 的阿里啊!!!)
10.接着在 ConfigurationPropertiesRebinder 中接收到这个事件, 触发 onApplicationEvent()
11.然后会执行 ConfigurationPropertiesRebinder#rebind()
12.其逻辑是将 postProcessBeforeInitialization 中存起来的 ConfigurationPropertiesBean 类型的 map 遍历, 进行重新绑定(即 destroyBean 后再 initializeBean, 则会从新配置重新赋值). 

总结: 有的地方(NacosContextRefresher)监听容器初始化添加配置更新的监听者, 自己不干活只发布配置刷新事件;

有的地方(ClientWorker)添加轮询任务获取服务端配置文件与本地 cacheData 对比 md5 来判断是否触发配置更新的监听者;

然后有的地方(RefreshEventListener)监听配置刷新事件然后用 SpringApplication.run() 通过触发 NacosPropertySourceLocator#locate() 为一个复制出来的 environment 对象添加从服务端获取最新配置, 再拷贝到当前的 environment 对象, 完成 environment 配置的刷新, 之后发布 environment 刷新事件;

有的地方(ConfigurationPropertiesRebinder)监听 environment 刷新事件, 然后为 @ConfigurationProperties 注解生成的 bean 重新绑定配置.

再总结: 挺好的, 奥运火炬手啊这是, 不停传递!!!!!

关键类

1.NacosConfigAutoConfiguration
    注册一个监听容器初始化事件的 bean (NacosContextRefresher)

2.NacosContextRefresher
    再初始化事件触发后添加一个 AbstractSharedListener 监听者到 cacheData

3.AbstractSharedListener
    当服务端配置与本地不同后触发
    触发后发布 RefreshEvent 事件

4.CacheData
    管理配置更新事件监听者
    对比服务端配置与本地配置的 md5
    md5 不同则触发 AbstractSharedListener
    
5.RefreshEvent
    通过容器发布
    当服务端配置发送更新触发
    触发后调用 ContextRefresher#refresh()
    
6.LongPollingRunnable
    轮询任务, 获取服务端最新配置, 与 cacheData 对比

7.ClientWorker
    管理轮询任务(LongPollingRunnable)
    负责与 nacos 服务端通信
    
8.ContextRefresher
    监听 RefreshEvent 事件
    负责获取服务端最新配置到 environment 对象中
    发布 EnvironmentChangeEvent 事件
    
9.EnvironmentChangeEvent
    通过容器发布
    当 environment 对象更新后触发
    触发后调用 ConfigurationPropertiesRebinder#rebind()
    
10.ConfigurationPropertiesRebinder
    监听 EnvironmentChangeEvent 事件
    负责重新绑定用了 @ConfigurationProperties 注解的 Bean 的值

graph TB

A1(NacosConfigAutoConfiguration)
A2(NacosContextRefresher)
A3(ApplicationReadyEvent)
A4(NacosContextRefresher#registerNacosListener)
A5(NacosConfigService#addListener)
 
A7(ClientWorker)
A8(LongPollingRunnable#轮询任务)
A9(NacosConfigService#getServerConfig)
B1(AbstractSharedListener#配置更新事件的监听者)
B2(cacheData)
B3(RefreshEvent)
B4(RefreshEventListener)
B5(ContextRefresher#refresh)
B6(ContextRefresher#refreshEnvironment)
B7(ContextRefresher#addConfigFilesToEnvironment)
B8(SpringApplication#run)
B9(StandardEnvironment)
C1(NacosPropertySourceLocator#locate)
C2(当前容器的 environment 对象)
C3(EnvironmentChangeEvent)
C4(ConfigurationPropertiesRebinder#onApplicationEvent)
C5(ConfigurationPropertiesRebinder#rebind)
C6(用了 ConfigurationProperties 注解 Bean)

A1--注册了一个-->A2
A2--监听了-->A3
A3--事件触发后调用-->A4
A4--通过-->A5
A5--注册了一个-->B1

 
A7--构造方法中创建了一个-->A8
A8--run 方法中调用-->A9
A9--获取最新配置与-->B2

B2--比较 md5, 不同则触发-->B1
B1--事件触发后发布-->B3
B4--监听了-->B3
B3--事件触发后调用-->B5
B5--调用-->B6
B6--调用-->B7
B7--利用-->B8
B7--复制当前容器 environment 对象到-->B9
B8--触发-->C1
C1--获取了最新配置到-->B9
B9--复制最新配置回到-->C2

C2--更新完配置后发布-->C3
G1(ConfigurationPropertiesRebinder)--监听了-->C3
C3--事件触发后调用-->C4
C4--调用-->C5
C5--获取新配置绑定到-->C6



image
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容