Apache ZooKeeper是一个分布式协调服务,它简化了分布式应用程序开发。Apache Hadoop、HBase等项目依赖Zookeeper,用于领导者选举、配置管理、节点协调、服务租约管理等。
ZooKeeper集群节点将数据存储在共享层级命名空间中,类似于标准的文件系统或树状数据结构。本文将展示如何使用Zookeeper的Java API来存储、更新和删除存储在Zookeeper中的信息。

Zookeeper安装和命令

官网下载最新版安装包,如apache-zookeeper-3.5.10-bin.tar.gz 。 为了简化,我们的示例使用单机版本,详细步骤参考安装文档。单机版本没有副本,如果Zookeeper处理异常,服务会停止。

单机环境很简单,解压即完成安装。使用下面命令启动Zookeeper服务:

./zkServer.sh start

输出内容如下:

ZooKeeper JMX enabled by default
Using config: E:\dev-tools\zookeeper\conf\zoo.cfg
Starting zookeeper ... STARTED

Zookeeper采用层级命名空间(类似于分布式文件系统)存储协调数据,如状态信息、协调信息、位置信息等。这些信息存储在不同节点上。在Zookeeper上的每个节点称为ZNode。zk中创建的节点分为两种:永久性节点和临时性节点。永久性节点即创建以后,在不执行delete命令的前提下,该节点是永久存在的;而临时节点与session有关,每个客户端与zk建立链接的时候会生成一个session,这个session不会因为链接zk服务器节点的变化而变化,只有当客户端断开连接以后,该session才会消失,而临时节点会随着session的消失而消失。

下面介绍常用的Zookeeper CLI示例。使用zkCli.sh 默认server为127.0.0.1:2181,在本机操作可以省略:zkCli.sh -server 127.0.0.1:2181。后面跟上要执行的命令。下面通过命令创建节点,然后获取节点的值。

  • 创建节点
.\bin\zkCli.cmd create /FirstNode v001

在命名空间的根节点上创建FirstNode节点,并设定值为v001.

  • 获取节点值
.\bin\zkCli.cmd get /FirstNode
  • 更新节点值
.\bin\zkCli.cmd set /FirstNode new001
  • 删除节点
.\bin\zkCli.cmd delete /FirstNode

Java Api 示例

首先加入依赖:

<dependency>
    <groupId>org.apache.zookeeper</groupId>
    <artifactId>zookeeper</artifactId>
    <version>3.4.11</version>
</dependency>

下面通过java API 创建节点、更新节点、返回数据。

ZooKeeper Java API主要由两个包组成:
org.apache.zookeeper:它定义了ZooKeeper客户端库的主类,以及很多ZooKeeper事件类型和状态的静态定义。
org.apache.zookeeper.data:它定义了与znode相关的特征,如访问控制列表(ACL)、id、统计信息等.

还有一些ZooKeeper Java api用于服务端实现,比如org.apache.zookeeper.server,org.apache.zookeeper.server.quorum,org.apache.zookeeper.server.upgrade。但这些超出了本文的范围。

连接Zookeeper实例

下面代码创建ZkConnection类,用于连接或关闭连接正在运行的Zookeeper服务:

public class ZKConnection {
    private ZooKeeper zoo;
    final CountDownLatch connectionLatch = new CountDownLatch(1);

    public ZooKeeper connect(String host) throws IOException, InterruptedException {
        zoo = new ZooKeeper(host, 2000, new Watcher() {
            @Override
            public void process(WatchedEvent we) {
                if (we.getState() == Event.KeeperState.SyncConnected) {
                    connectionLatch.countDown();
                }
            }
        });
        connectionLatch.await();
        return zoo;
    }

    public void close() throws InterruptedException {
        zoo.close();
    }
}

要使用Zookeeper服务,应用程序首先需要实例化ZooKeeper 类,它是Zookeeper客户端的主类。在connect方法中,实例化ZooKeeper类,为了保证连接成功,注册了回调方法,处理从Zookeeper服务端传回的连接确认事件WatchedEvent ,方法内使用CountDownLatch的countdown方法,用于解除阻塞并返回连接实例。一旦建立到服务器的连接,给客户端分配会话ID。为了保持会话有效,客户端应该定期向服务器发送心跳。只要它的会话ID仍然有效,则客户端可以调用ZooKeeper api。

客户端操作

首先定义操作接口,用于操作ZNode,如创建节点、更新节点值、获取节点值:

import org.apache.zookeeper.KeeperException;

public interface ZKManager {
    /**
     * Create a Znode and save some data
     *
     * @param path path
     * @param data data
     */
    void create(String path, byte[] data) throws KeeperException, InterruptedException;

    /**
     * Get ZNode Data
     *
     * @param path path
     * @param  watchFlag flag
     */
    Object getZNodeData(String path, boolean watchFlag);

    /**
     * Update the ZNode Data
     *
     * @param path path
     * @param data data
     */
    void update(String path, byte[] data) throws KeeperException, InterruptedException, KeeperException;
}

接口实现类:

public class ZKManagerImpl implements ZKManager {
    private static ZooKeeper zkeeper;
    private static ZKConnection zkConnection;

    public ZKManagerImpl() {
        initialize();
    }

    /** * Initialize connection */
    private void initialize() {
        try {
            zkConnection = new ZKConnection();
            zkeeper = zkConnection.connect("localhost");
        } catch (Exception e) {
            System.out.println(e.getMessage());
        }
    }

    public void closeConnection() {
        try {
            zkConnection.close();
        } catch (InterruptedException e) {
            System.out.println(e.getMessage());
        }
    }

    @Override
    public void create(String path, byte[] data) throws KeeperException, InterruptedException {
        zkeeper.create(path, data, ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
    }

    @Override
    public Object getZNodeData(String path, boolean watchFlag) {
        try {
            byte[] b;
            b = zkeeper.getData(path, null, null);
            String data = new String(b, StandardCharsets.UTF_8);
            System.out.println(data);
            return data;
        } catch (Exception e) {
            System.out.println(e.getMessage());
        }
        return null;
    }

    @Override
    public void update(String path, byte[] data) throws KeeperException, InterruptedException {
        int version = zkeeper.exists(path, true)
                .getVersion();
        zkeeper.setData(path, data, version);
    }
}

上面代码中,connect 和 disconnect方法调用之前创建ZKConnection类的实例。create方法创建ZNode节点,参数为路径以及内容对应的字节数组,为了演示我们没有使用ACL,且为持久节点,即使客户端断开连接节点也不会被删除。

getZNodeData方法从ZooKeeper中获取ZNode数据的逻辑非常简单。最后使用update方法,检查ZNode在给定路径是否存在。除此之外,为了更新数据首先检查ZNode是否存在并获得当前版本。然后用ZNode路径、data和当前版本作为参数调用setData方法。只有传入的版本与最新版本匹配,ZooKeeper才会更新数据。

总结

在开发分布式应用程序时,Apache ZooKeeper作为分布式协调服务起着至关重要的作用。特别是对于存储共享配置、选择主节点等应用场景。ZooKeeper还提供了优雅的javaAPI,用于客户端应用程序代码与ZooKeeper znode间无缝通信。