zookeeper-05-javaAPI使用

[TOC]

本文主要记录自己学习zookeeper时的一些个人笔记。不喜勿喷。

1 环境准备

随便建个java项目即可。

maven坐标:

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

2 同步方式调用

public class ZKTestSync {

    public static String connectStr = "192.168.161.128:2181";
    public ZooKeeper keeper = null;
    private static Stat stat = new Stat();
    private static Logger log = LoggerFactory.getLogger(ZKTestSync.class);

    @Before
    public void init() {
        try {
            keeper = new ZooKeeper(connectStr, 3000, (WatchedEvent event) -> {
                log.info("事件:{}", event);
                if (event.getState() == KeeperState.SyncConnected) {
                    if (event.getType() == EventType.None && event.getPath() == null) {

                    } else {
                        if (event.getType() == EventType.NodeChildrenChanged) {
                            log.info("节点{}发生变化", event.getPath());
                        } else if (EventType.NodeDataChanged == event.getType()) {
                            try {
                                log.info("节点{}数据发生变化", event.getPath());
                                this.keeper.getData(event.getPath(), false, stat);
                            } catch (Exception e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            });
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @After
    public void close() {
        if (keeper != null)
            try {
                keeper.close();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
    }

    @Test
    public void testCreateNode() {
        try {
            String string = this.keeper.create("/node_3", "3".getBytes(), Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
            System.out.println(string);
        } catch (KeeperException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testGetChildren() {
        try {
            List<String> list = this.keeper.getChildren("/", false);
            list.stream().forEach(System.out::println);

            // true表示对节点的变化感兴趣,
            // 在节点变化时在ZooKeeper构造函数传入的Watcher中可收到通知.
            // 但是这种监听器只是一次性的
            list = this.keeper.getChildren("/", true);
            list.stream().forEach(System.out::println);
            Thread.sleep(3 * 60 * 1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testGetData() {
        try {
            String str = new String(this.keeper.getData("/node_8", true, stat));
            System.out.println(str);
            Thread.sleep(3 * 60 * 1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testDelete() {
        try {
            this.keeper.delete("/node_40000000005", -1);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (KeeperException e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testSetData() {
        try {
            Stat s = this.keeper.setData("/node_1", "ddd".getBytes(), -1);
            System.out.println(s);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testSetACL() {
        try {
            ACL ipAcl = new ACL(Perms.CREATE | Perms.DELETE | Perms.READ, new Id("ip", "192.168.161.1"));
            ACL digestAcl = new ACL(Perms.READ | Perms.WRITE, new Id("digest", DigestAuthenticationProvider.generateDigest("hylexus:123456")));
            List<ACL> acls = Arrays.asList(ipAcl, digestAcl);
            String string = this.keeper.create("/node_11", "8".getBytes(), acls, CreateMode.PERSISTENT);
            System.out.println(string);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testGenerateDigest() throws NoSuchAlgorithmException {
        System.out.println(DigestAuthenticationProvider.generateDigest("hylexus:123"));
    }

}

3 异步方式调用

和上面的同步代码的最大区别就是,异步代码没法及时获取返回值,或者说他没有返回值。
只能通过提供回调函数的方式来处理操作完成后的工作。
但是有了回调函数,难免代码中有大量匿名类,为简单,此处使用java8的lambda代替匿名类。

public class ZKTestASync {

    public static String connectStr = "192.168.161.128:2181";
    public ZooKeeper keeper = null;
    private static final Logger log = LoggerFactory.getLogger(ZKTestASync.class);
    private static Stat stat = new Stat();

    @Before
    public void init() {
        try {
            keeper = new ZooKeeper(connectStr, 3000, (WatchedEvent event) -> {
                log.info("事件:{}", event);
                if (event.getState() == KeeperState.SyncConnected) {
                    if (event.getType() == EventType.None && event.getPath() == null) {

                    } else {
                        if (event.getType() == EventType.NodeChildrenChanged) {
                            log.info("节点{}发生变化", event.getPath());
                        } else if (EventType.NodeDataChanged == event.getType()) {
                            try {
                                log.info("节点{}数据发生变化", event.getPath());
                                this.keeper.getData(event.getPath(), false, stat);
                            } catch (Exception e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            });
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @After
    public void close() {
        if (keeper != null)
            try {
                keeper.close();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
    }

    // 异步创建
    @Test
    public void testCreateNodeASync() {
        // 异步调用,没有返回值,通过回调函数处理结果
        this.keeper.create("/node_4", "3".getBytes(), Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_SEQUENTIAL, new StringCallback() {
            /***
             * @param rc:返回码
             * @param path:要创建的节点的完整路径(想要创建的路径)
             * @param ctx:create方法传入的上下文参数,此处是
             *            "testCreateNodeASync"
             * @param name:返回的创建的真实路径(创建顺序节点时返回的真实路径和传入的路径是不同的)
             */
            @Override
            public void processResult(int rc, String path, Object ctx, String name) {
                log.info("rc:{}", rc);
                log.info("path:{}", path);
                log.info("ctx:{}", ctx);
                log.info("name:{}", name);
            }
        }, "testCreateNodeASync");
    }

    @Test
    public void testGetChildren() {
        try {
            /***
             * @param rc:返回码
             * @param path:要创建的节点的完整路径(想要创建的路径)
             * @param ctx:create方法传入的上下文参数
             * @param children:子节点列表
             * @param stat:节点状态
             */
            this.keeper.getChildren("/", true, //
                    (int rc, String path, Object ctx, List<String> children, Stat stat) -> {
                        log.info("rc:{}", rc);
                        log.info("path:{}", path);
                        log.info("ctx:{}", ctx);
                        log.info("children:{}", children);
                        log.info("stat:{}", stat);
                    }, "这里可以传入任何Object作为上下文以便在回调函数函数中使用");
            Thread.sleep(3 * 60 * 100);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testGetData() {
        try {
            this.keeper.getData("/node_1", true, (int rc, String path, Object ctx, byte[] data, Stat stat) -> {
                log.info("rc:{}", rc);
                log.info("path:{}", path);
                log.info("ctx:{}", ctx);
                log.info("data:{}", new String(data));
                log.info("stat:{}", stat);
            }, null);
            Thread.sleep(3 * 60 * 1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testDelete() {
        try {
            this.keeper.delete("/node_6", -1, (int rc, String path, Object ctx) -> {
                log.info("rc:{}", rc);
                log.info("path:{}", path);
                log.info("ctx:{}", ctx);
            }, null);
            Thread.sleep(3 * 60 * 1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testSetData() {
        try {
            this.keeper.setData("/node_1", "aaa".getBytes(), -1, (int rc, String path, Object ctx, Stat stat) -> {
                log.info("rc:{}", rc);
                log.info("path:{}", path);
                log.info("ctx:{}", ctx);
                log.info("stat:{}", stat);
            }, null);
            Thread.sleep(3 * 60 * 1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

4 总结

  • 整个API还是简单易用的
  • 注册的监听器只是一次性的,没有提供类似于自动注册多次的API
  • session的超时重连可能导致watcher的重复执行,需要手动自己控制
  • 返回值中还有的是byte[],入参也有byte[]。操作不是很舒服

当然,也有zookeeper编程的其他框架可用,比如ZkClient等

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,633评论 18 139
  • 一、ZooKeeper的背景 1.1 认识ZooKeeper ZooKeeper---译名为“动物园管理员”。动物...
    algernoon阅读 9,058评论 1 106
  • 用消费,麻痹自己
    小陆_10a4阅读 110评论 0 0
  • 在静夜里随想 内心的波澜 时而起伏时而跌宕 兀自反问 如果屈原没有投江 流水会不会呜咽 山河会不会沦丧 历史会不会...
    安言靜语阅读 512评论 0 1
  • 第二周的思考与分享 上周二的课上,刘老师讲,静心就是检索自己,看自己处于什么状态,后面讲能量层级,吧啦吧啦讲了好多...
    巨晓松阅读 692评论 0 1