一、设计
例如,使用路径为/queue的znode下的节点表示队列中的元素。/queue下的节点都是顺序持久化znode。这些znode名字的后缀数字表示了对应队列元素在队列中的位置。znode名字后缀数字越小,对应队列元素在队列中的位置越靠前。例如:
data:image/s3,"s3://crabby-images/a2a9b/a2a9b3df084a62228a9927400c320eb46dc57acf" alt=""
二、主要方法
data:image/s3,"s3://crabby-images/1524b/1524bebc2ebee603fad17c52e6677a2518424b19" alt=""
- 入队
03 offer方法.png
在一个死循环里面,调用zookeeper.create方法,创建PERSISTENT_SEQUENTIAL模式的数据,其前缀为qn- - 出队
04 element方法.png
在不更改队列的条件下,出队队列快照中的数据。注意的是getData的成功执行并不意味着出队成功,原因是可能别其他客户端给出队了。如果读取到了队列为空的状态的话,会抛出异常
NoSuchElementException。也正是因为Since other clients are remove()ing and take()ing nodes concurrently,所以把操作都是放于一个死循环(while(true))中执行。 - 删除
05 remove方法.png
与element()不同的地方为调用了一次删除操作:zookeeper.delete(path, -1); -
出队并移除,并且确保能够返回并移除成功。
06 take方法.png
07 LatchChildWatcher方法.png
08 orderredChildren方法.png
通过一个LatchChildWatcher来监控目录数据的变动。并确保每次能出队一个数据。与与element()方法最明显的不同之处为,不会抛出异常NoSuchElementException。
三、执行测试
通过类org.apache.zookeeper.recipes.queue.DistributedQueueTest类执行测试案例。
因为此类继承自org.apache.zookeeper.test.ClientBase,在ClientBase类中,有几个重要的初始化变量需要配置,如下图:
data:image/s3,"s3://crabby-images/df1c7/df1c77099e4a2f1428c923d11a9e1d89401c25ef" alt=""
如果变量BASETEST取值与系统参数"build.test.dir",所以在IDE中,需要 设置测试运行的JVM运行参数 -Dbuild.test.dir=D:/study/zookeeper/source/dataDir,
data:image/s3,"s3://crabby-images/52d37/52d375d79986681dfc555031fc6e3a614c4a39ef" alt=""
否则,会报以下错误:
data:image/s3,"s3://crabby-images/cd29b/cd29bf2e61a2eecfd4f361b198e08c8358ae0088" alt=""
启动ZookeeperServerMain,并执行测试案例:
data:image/s3,"s3://crabby-images/5455e/5455e0983846b6c9bbf89ea1b076c54abbbb7d76" alt=""
网友评论