前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Zookeeper实现微服务统一配置中心

Zookeeper实现微服务统一配置中心

作者头像
阿凯
发布2019-12-19 11:21:27
1.5K0
发布2019-12-19 11:21:27
举报
文章被收录于专栏:程序员阿凯程序员阿凯

Zookeeper:它是一个分布式服务框架,是Apache Hadoop 的一个子项目,提供一种集中式信息存储服务,用于解决分布式应用中的一些数据管理问题

  • 简单来说: zookeeper = 文件系统 + 监听通知机制

作用:

实现分布式统一配置中心

服务注册中心

分布式锁等功能

用于分布式应用程序的高性能协调

ZK实现配置中心操作

ZK 配置节点信息

我们通过 zkui [https://github.com/DeemOpen/zkui]添加一个节点,并存放一些数据信息

Java 实现

1.pom 引入

代码语言:javascript
复制
<parent>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-parent</artifactId>
 <version>2.0.5.RELEASE</version>
</parent>

<dependencies>
 <dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-web</artifactId>
 </dependency>

 <dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-configuration-processor</artifactId>
 <optional>true</optional>
 </dependency>

 <dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-test</artifactId>
 </dependency>

 <dependency>
 <groupId>junit</groupId>
 <artifactId>junit</artifactId>
 <version>4.12</version>
 </dependency>
 
 <dependency>
 <groupId>org.apache.curator</groupId>
 <artifactId>curator-recipes</artifactId>
 <version>4.0.0</version>
 </dependency>
</dependencies>

<properties>
 <java.version>1.8</java.version>
</properties>

2.配置中心 本地配置文件 application.properties 中,需要配置关于Zookeeper配置中心的信息

代码语言:javascript
复制
config.zookeeper.url=127.0.0.1:2181
config.zookeeper.nodename=pay-server-config

3. 测试用例

代码语言:javascript
复制
    @RunWith(SpringRunner.class)
    @SpringBootTest
    public class ZKTest {
    
        @Value("${config.zookeeper.url}")
        String zkUrl;   // 配置文件读取zk的地址
    
        @Value("${config.zookeeper.nodename}")
        String nodename;  // 配置所在的节点名
    
        @Test
        public void test() throws Exception{
            CuratorFramework zkclient = CuratorFrameworkFactory.newClient(zkUrl,
                    new RetryOneTime(1000));
            // 启动和zookeeper的连接
            zkclient.start();
    
            // 1. 获取节点
            byte[] bytes = zkclient.getData().forPath("/" + nodename);
            
            // 2. 获取节点下的配置
            List<String> strings = zkclient.getChildren().forPath("/" + nodename);
            strings.forEach(s -> {
                System.out.println(s);
            });
        }
    }

4. 监听配置修改

当配置中心的数据发生修改,我们需要实时的更新到各个站点去

Zookeeper的订阅发布

代码语言:javascript
复制
zk的发布订阅也就是watch机制,是一个轻量级的设计。

它采用了一种推拉模式。一旦服务端感知数据发生了变化,那么会发送一个事件类型和节点信息给关注的客户端,而不会包括具体的变更内容,所以事件本身是轻量级的,这就是所谓的“推”部分。然后,收到变更通知的客户端需要自己去拉变更的数据,这就是“拉”部分。

img

Java客户端实现

启动监听:

代码语言:javascript
复制
// 监听配置发生修改
TreeCache treeCache = new TreeCache(zkclient, "/" + nodename);
treeCache.start(); // 监听整个节点内的数据变化

treeCache.getListenable().addListener(new TreeCacheListener() {
    
    @Override
	public void childEvent(CuratorFramework framework, TreeCacheEvent event) throws Exception {
        // TODO 如果zk中有数据变化,执行相应的代码

    }
});

事件类型:

代码语言:javascript
复制
/**
 *  Type of change
 */
public enum TreeCacheEvent.Type {

        NODE_ADDED,   // 添加子节点
        NODE_UPDATED, // 节点数据发生修改
        NODE_REMOVED, // 节点被移除

        CONNECTION_RECONNECTED, // ZK挂掉
        CONNECTION_LOST,     // 重新启动ZK
        INITIALIZED// ZK挂掉一段时间后初始化操作
    }

获取key,value :

代码语言:javascript
复制
@Configuration
public class ZkConfig {

    @Value("${config.zookeeper.url}")
    String zkUrl;   // 配置文件读取zk的地址
    @Value("${config.zookeeper.nodename}")
    String nodename;  // 配置所在的节点名

    private Properties properties = new Properties();

    public String getProperty(String key) {
        String value = properties.getProperty(key);

        if (StringUtils.isEmpty(value)) {
            throw new RuntimeException("配置信息不存在");
        }
        return value;
    }

    @PostConstruct
    public void init() {
        CuratorFramework zkclient = CuratorFrameworkFactory.newClient(zkUrl,
                new RetryOneTime(1000));
        zkclient.start();// 启动Zookeeper的连接

        try {
            // 1. 读取zookeeper数据
            HashMap<String, String> configMap = new HashMap<>();
            List<String> configKeys = zkclient.getChildren().forPath("/" + nodename);
            for (String configKey : configKeys) {
                byte[] data = zkclient.getData().forPath("/" + nodename + "/" + configKey);
                configMap.put(configKey, new String(data));
            }

            // 2. 把数据存储到 properties中
            properties.putAll(configMap);

            // 3. 配置发生修改  -->  watch监听机制
            TreeCache treeCache = new TreeCache(zkclient, "/" + nodename);
            treeCache.start(); // 监听整个节点内的数据变化

            treeCache.getListenable().addListener(new TreeCacheListener() {
                @Override
                public void childEvent(CuratorFramework curatorFramework, TreeCacheEvent treeCacheEvent) throws Exception {
                    // 如果zk中有数据变化,执行相应的代码
                    switch (treeCacheEvent.getType()) {
                        case NODE_UPDATED:
                            System.out.println("数据发生了变化: "+treeCacheEvent.getData());

                            String key = treeCacheEvent.getData().getPath().replace("/"+nodename+"/","");
                            String value = new String(treeCacheEvent.getData().getData());
                            properties.setProperty(key, value);
                            break;
                        default:
                            break;
                    }
                }
            });

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-12-15,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 程序员阿凯 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • ZK实现配置中心操作
相关产品与服务
微服务引擎 TSE
微服务引擎(Tencent Cloud Service Engine)提供开箱即用的云上全场景微服务解决方案。支持开源增强的云原生注册配置中心(Zookeeper、Nacos 和 Apollo),北极星网格(腾讯自研并开源的 PolarisMesh)、云原生 API 网关(Kong)以及微服务应用托管的弹性微服务平台。微服务引擎完全兼容开源版本的使用方式,在功能、可用性和可运维性等多个方面进行增强。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档