Zookeeper观察者

public class TestZKWatch {

    private String connectString="hadoop101:2181,hadoop102:2181";
    private int sessionTimeout=6000;
    private ZooKeeper zooKeeper;

    //1.创建客户端  zkCli.sh -server xxx:2181  
    @Before
    public void init() throws Exception {
         
        // 创建一个zk的客户端对象
         zooKeeper = new ZooKeeper(connectString, sessionTimeout, new Watcher() {
            //回调方法,一旦watcher观察的path触发了指定的事件,服务端会通知客户端,客户端收到通知后
            // 会自动调用process()
            @Override
            public void process(WatchedEvent event) {
                //System.out.println(event.getPath()+"发生了以下事件:"+event.getType());
                //重新查询当前路径的所有新的节点             
            }
        });
        System.out.println(zooKeeper);
    }
    
    
    @After
    public void close() throws InterruptedException {        
        if (zooKeeper !=null) {
            zooKeeper.close();
        }        
    }
    
    

    //2.ls设置观察者
    //注意:getChilden(path, true);
    //调用默认的观察者watch方式,但每个函数应该调用不同的watch,所以一般都是自己编写watch观察者
    
    // ls path  watch
    @Test
    public void lsAndWatch() throws Exception {      
        //传入true,默认使用客户端自带的观察者 
        zooKeeper.getChildren("/data2",new Watcher() {            
            //当前线程自己设置的观察者
            @Override
            public void process(WatchedEvent event) {                
                System.out.println(event.getPath()+"发生了以下事件:"+event.getType());
                
                List children;
                try {
                    children = zooKeeper.getChildren("/data2", null);
                    System.out.println(event.getPath()+"的新节点:"+children);
                } catch (KeeperException e) {
                    e.printStackTrace();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }                
            }
        });
        
        //客户端所在的进程不能死亡
        while(true) {           
            Thread.sleep(5000);            
            System.out.println("我还活着......");            
        }        
    }
    

    //3.get设置监听
    private CountDownLatch cdl=new CountDownLatch(1)  
    // 监听器的特点: 只有当次有效
    // get path watch
    @Test
    public void getAndWatch() throws Exception {
        //zooKeeper = new ZooKeeper(connectString, sessionTimeout, true)  会自动调用Before里面的process
        //是Connect线程调用
        byte[] data = zooKeeper.getData("/data2", new Watcher() {            
            // 是Listener线程调用
            @Override
            public void process(WatchedEvent event) {                
                System.out.println(event.getPath()+"发生了以下事件:"+event.getType());

                //减一
                cdl.countDown();
            }
        }, null);        
        System.out.println("查询到的数据是:"+new String(data));   
        //阻塞当前线程,当初始化的值变为0时,当前线程会唤醒
        cdl.await();                     
    }
    

    //4.设置持续监听:解决只能监听一次,在监听回调process中设置新的观察者

        //1. 持续watch:  不合适
        @Test
        public void lsAndAlwaysWatch() throws Exception {            
            //传入true,默认使用客户端自带的观察者
            zooKeeper.getChildren("/data2",new Watcher() {                
                // process由listener线程调用listener线程不能阻塞,阻塞后无法再调用process
                //当前线程自己设置的观察者
                @Override
                public void process(WatchedEvent event) {                    
                    System.out.println(event.getPath()+"发生了以下事件:"+event.getType());                   
                    System.out.println(Thread.currentThread().getName()+"---->我还活着......");                   
                    try {
                        lsAndAlwaysWatch();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }                 
                }
            });
            
            //客户端所在的进程不能死亡---listener在每间隔5秒,阻塞了不能调用process,listener是不能阻塞的
            while(true) {                
                Thread.sleep(5000);                
                System.out.println(Thread.currentThread().getName()+"---->我还活着......");                
            }            
        }
        
        // 2.持续watch:  合适
        @Test
        public void testLsAndAlwaysWatchCurrent() throws Exception {                                    

                   lsAndAlwaysWatchCurrent();                    
                    //客户端所在的进程不能死亡
                    while(true) {                        
                        Thread.sleep(5000);                        
                        System.out.println(Thread.currentThread().getName()+"---->我还活着......");                        
                    }                    
        }
        
        @Test
        public void lsAndAlwaysWatchCurrent() throws Exception {            
            //传入true,默认使用客户端自带的观察者
            zooKeeper.getChildren("/data2",new Watcher() {                
                // process由listener线程调用listener线程不能阻塞,阻塞后无法再调用process
                //当前线程自己设置的观察者
                @Override
                public void process(WatchedEvent event) {                    
                    System.out.println(event.getPath()+"发生了以下事件:"+event.getType());                    
                    System.out.println(Thread.currentThread().getName()+"---->我还活着......");                    
                    try {
                        //递归调用
                        lsAndAlwaysWatchCurrent();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }                    
                }
            });
        }

}

你可能感兴趣的:(Zookeeper)