14 May 2024 |
- DistributedLocks实现
- 自动注入数据源。创建连接,保证不影响其他连接。
- locked原子变量判断是否加锁成功;
- Executor 定时任务,尝试获取锁。
data:image/s3,"s3://crabby-images/b836f/b836f90b38763bcc4b584f10075b986550ed8192" alt="图片"
- init()初始化方法
使用注解@PostConstruct初始化,使用数据源创建连接。
开启定时任务,每隔5秒执行一次。
data:image/s3,"s3://crabby-images/5624e/5624e49b4153e2166084885ecf421c175e2f73d4" alt="图片"
- tryLock()方法
尝试获取锁,获取成功,locked=true,否则为false。
data:image/s3,"s3://crabby-images/e627e/e627ed902b5f6b109664584b58af84d41589d766" alt="图片"
- lock() 加锁
- 关闭自动提交事务;
- 设置事务隔离级别为RC;
- 设置锁超时时间innodb_lock_wait_timeout为5秒;
- 执行锁定语句for update加锁
- 事务不要commit或rollback,否则锁就释放了
- 根据locked状态判断是第一次获取到锁,还是后续的锁重入。
data:image/s3,"s3://crabby-images/b8ee4/b8ee4c2cd270662f832e7dfd4d33833e6632b28d" alt="图片"
- closed()方法
使用@PreDestroy注解销毁bean时,连接回滚,并关闭。
data:image/s3,"s3://crabby-images/d6bb6/d6bb652d262c5d0801d16edfe643292372950f8b" alt="图片"
- 测试
- 分别用端口9129和9130启动config-server
假如9129先启动,观察日志成功获取到锁了,并在定时任务重不断重入。
data:image/s3,"s3://crabby-images/b3f7c/b3f7cdc9ba9393c2fc5f3b57b81f821f8e16432e" alt="图片"
- 9130端口后启动,不会获取到锁,等待5秒后,抛出超时异常。
data:image/s3,"s3://crabby-images/8d7d7/8d7d7e867bda24edbcd371f54571a29a6ab19162" alt="图片"
data:image/s3,"s3://crabby-images/7734a/7734a44242fc503a1cd8151baf5d777561d6ad6e" alt="图片"
锁状态locked表示已经持有
data:image/s3,"s3://crabby-images/08964/08964707d926bc889ec52491907a4040a7f266e0" alt="图片"
源码:https://github.com/midnight2104/midnight-config/tree/v5
10 May 2024 |
- 定义SpringValueProcessor处理类
- 实现BeanPostProcessor后置处理器接口,扫描所有的Spring value,保存起来。
- 实现ApplicationListener接口,在配置变更时,更新所有的spring value
data:image/s3,"s3://crabby-images/4e085/4e085bb9e2d9b9f79c4a76b8ec0e48c1123fba8c" alt="图片"
- 实现BeanPostProcessor后置处理器接口
实现postProcessBeforeInitialization()方法。通过工具类找到所有带有@Value的bean,遍历每个字段。通过helper工具类抽取注解上面的表达式,构建SpringValue对象并保存起来。
data:image/s3,"s3://crabby-images/f8192/f8192c4f11ddac0717481f3fcfc52ce7add5f1bc" alt="图片"
使用SpringValue对象封装@Value注解信息。
data:image/s3,"s3://crabby-images/9e9dc/9e9dc4d1dcbcdd3c8daab850c41de45a6c209198" alt="图片"
一个SpringValue对象如下:
data:image/s3,"s3://crabby-images/18ad7/18ad7229ec157eb0776114e3aea5034562d75584" alt="图片"
- 实现onApplicationEvent方法
- 事件变更时,自动实现该方法;
- 从前面保存的holder中获取SpringValues;
- 遍历集合,解析获取属性值;
- 通过反射重新赋值。
data:image/s3,"s3://crabby-images/34805/3480551cf16cb40e7a004b6118b2b123bca35957" alt="图片"
- 测试
- 启动 config-server
- 启动config-client
初始值如下:都是dev100
data:image/s3,"s3://crabby-images/c37f9/c37f993a2530527393cb694718dbd031e08e197e" alt="图片"
更新配置中心的值
data:image/s3,"s3://crabby-images/c2c85/c2c852c0009dd1c5701ea780401e536c41d6c509" alt="图片"
观察config-demo的日志
data:image/s3,"s3://crabby-images/e6f8a/e6f8aa0455194910d6dec89a6aa3e0128d4119c2" alt="图片"
再次获取值,就已经动态发生了变化。
data:image/s3,"s3://crabby-images/8ada7/8ada7462ad3ec64d26e2f9da1bdcd26b399da39a" alt="图片"
源码:https://github.com/midnight2104/midnight-config/tree/v4
09 May 2024 |
- 在PropertySourcesProcessor中,需要通过http从config-server获取配置。
data:image/s3,"s3://crabby-images/31115/31115c78019f4f6276cf02f8c1e9513904eb5eeb" alt="图片"
- 使用ConfigMeta包装服务信息
data:image/s3,"s3://crabby-images/b133f/b133f42b198d9ae47e6f282865155e4976d69d9d" alt="图片"
- 在MidnightConfigService接口中添加默认实现类
- 继承MidnightRepositoryChangeListener接口;
- 获取默认的MidnightRepository;
- 创建MidnightConfigServiceImpl实现类;
- 添加listener
data:image/s3,"s3://crabby-images/32a76/32a765be8a86ff6c2a6e69cc85d5fac62cbfec6e" alt="图片"
- 定义MidnightRepositoryChangeListener,用于事件监听的回调接口
- onChange() 执行方法
- ChangeEvent变动事件,传递参数
data:image/s3,"s3://crabby-images/eb4d6/eb4d631db3cb0c6a498739d8b326c7ef75f150a1" alt="图片"
- 定义MidnightRepository接口
- 提供默认实现类
- getConfig()获取配置
- addListener()添加监听器
data:image/s3,"s3://crabby-images/c092c/c092ce5af7903e40e44018df984d40c920ffea08" alt="图片"
- MidnightRepositoryImpl实现从server获取配置
- meta保存服务配置源信息;
- versionMap保存server版本号,用于比较本地和远程版本信息,用于比对配置是否发生变化;
- configMap保存一个ns服务下所有配置信息;
- executor定时任务,监听服务端配置是否发生变化;
- listeners保存监听器。
data:image/s3,"s3://crabby-images/22464/2246433c92b0a8e22479ea533eed54fa1e4ac844" alt="图片"
- findAll()接口
发起http请求,从config-server获取所有配置信息。
data:image/s3,"s3://crabby-images/22592/225928a0627fb83fb7a2aaaba40ff63412dcddb4" alt="图片"
- heartbeat() 心跳检测
- 发起http请求,从配置中心服务端获取最新版本信息;
- 当远程版本信息大于本地版本信息时,说明配置发生了变化。
- 保存最新版本信息;
- 从配置中心重新获取所有配置信息;
- 保存最新的配置信息;
- 发布配置发生变更事件
data:image/s3,"s3://crabby-images/bfe2c/bfe2ca0972de4932e3540d2bae4a9981fcf3b792" alt="图片"
- onChange() 发布事件
- 更新本地最新配置
- 通过Spring发布环境变更事件。事件类EnvironmentChangeEvent来自于SpringCloud,解决微服务架构下配置热更新问题。
data:image/s3,"s3://crabby-images/b956c/b956c26239ab3e7b9b6ba739b17be1b087125c61" alt="图片"
- 重新绑定配置
当EnvironmentChangeEvent事件发布之后,会重新绑定PropertySource,调用getProperty()方法获取最新配置。
data:image/s3,"s3://crabby-images/3bcb5/3bcb583591659176c5a940b94b4f7bcfcca9bfef" alt="图片"
- @ConfigurationProperties 中的属性就会被更新
data:image/s3,"s3://crabby-images/55da9/55da93d5a95bf0b9ab89985fc9d8d79da4f48523" alt="图片"
- 测试
- 启动config-server
- 启动config-client
- 观察日志,最开始的值如下
data:image/s3,"s3://crabby-images/21364/21364361985920c4b0cb41d2ab4f09a5753c74fe" alt="图片"
发现版本发生了变化,重新获取了配置,并发布了EnvironmentChangeEvent事件。
data:image/s3,"s3://crabby-images/58df7/58df73e00e4d8b235e136f08e84f06562da5425c" alt="图片"
data:image/s3,"s3://crabby-images/9edc4/9edc480bcfdaa00b3f06fb0e1b10f50d28c668ab" alt="图片"
源代码:
https://github.com/midnight2104/midnight-config/tree/v3