一种基于zookeeper的分布式队列的设计与实现
package com.ysl.zkclient.queue; import com.ysl.zkclient.ZKClient;
import com.ysl.zkclient.exception.ZKNoNodeException;
import com.ysl.zkclient.utils.ExceptionUtil;
import org.apache.zookeeper.CreateMode;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import java.io.Serializable;
import java.util.List; /**
* 一种分布式队列的实现
* @param <T>
*/
public class ZKDistributedQueue<T extends Serializable> { private static final Logger LOG = LoggerFactory.getLogger(ZKDistributedQueue.class); private static final String ELEMENT_NAME = "node"; private ZKClient client;
private String rootPath; /**
* 创建分布式队列
* @param client zk客户端
* @param rootPath 队列的跟路径
*/
public ZKDistributedQueue(ZKClient client, String rootPath) {
this.client = client;
this.rootPath = rootPath;
if(!client.exists(rootPath)){
throw new ZKNoNodeException("the root path is not exists, please create path first ["+rootPath+"]");
}
} /**
* 添加一个元素
* @param node
* @return
*/
public boolean offer(T node){
try{
client.create(rootPath+"/"+ELEMENT_NAME + "-",node, CreateMode.PERSISTENT_SEQUENTIAL);
}catch (Exception e){
throw ExceptionUtil.convertToRuntimeException(e);
}
return true;
} /**
* 删除并返回顶部元素
* @return
*/
public T pool(){
while(true){
Node node = getFirstNode();
if(node == null){
return null;
} try{
boolean flag = client.delete(node.getName());
if(flag){
return (T)node.getData();
}else{
//删除失败,说明数据已经被其他的线程获取,重新获取底部元素
}
}catch (Exception e){
throw ExceptionUtil.convertToRuntimeException(e);
}
}
} /**
* 获取队列顶部元素
* @return
*/
private Node<T> getFirstNode() {
try{
while(true){
List<String> children = client.getChild(rootPath,true);
if(children == null || children.isEmpty()){
return null;
} String nodeName = getNodeName(children);
try{
return new Node<T>(rootPath+"/"+nodeName,(T)client.getData(rootPath+"/"+nodeName));
}catch (ZKNoNodeException e){
//如果抛出此异常,证明该节点已被其他线程获取
}
}
}catch (Exception e){
throw ExceptionUtil.convertToRuntimeException(e);
}
} /**
* 获取编号最小的节点
* @param children
* @return
*/
private String getNodeName(List<String> children) {
String child= children.get(0);
for(String path : children){
if(path.compareTo(child) < 0){
child = path;
}
}
return child;
} public boolean isEmpty(){
return client.getChild(rootPath,true).size() == 0;
} public T peek(){
Node<T> node = getFirstNode();
if(node == null){
return null;
}
return node.getData();
} private class Node<T>{ private String name;
private T data; public Node(String name, T data) {
this.name = name;
this.data = data;
} public String getName() {
return name;
} public T getData() {
return data;
}
}
}
测试代码
/**
* 测试分布式队列
* @throws Exception
* @return void
*/
@Test
public void testDistributedQueue() throws Exception{
final String rootPath = "/zk/queue";
//创建rootPath
zkClient.createRecursive(rootPath, null, CreateMode.PERSISTENT); final List<String> list1 = new ArrayList<String>();
final List<String> list2 = new ArrayList<String>();
for(int i=0;i<21;i++){
Thread thread1 = new Thread(new Runnable() {
public void run() {
ZKDistributedQueue<String> queue = new ZKDistributedQueue(zkClient, rootPath);
queue.offer(Thread.currentThread().getName());
list1.add(Thread.currentThread().getName());
}
});
thread1.start();
} //等待事件到达
int size1 = TestUtil.waitUntil(21, new Callable<Integer>() {
@Override
public Integer call() throws Exception {
return list1.size();
} }, TimeUnit.SECONDS, 100);
System.out.println(zkClient.getChildren(rootPath)); for(int i=0;i<20;i++){
Thread thread = new Thread(new Runnable() {
public void run() {
ZKDistributedQueue<String> queue = new ZKDistributedQueue(zkClient, rootPath);
list2.add(queue.poll());
}
});
thread.start();
}
//等待事件到达
int size2 = TestUtil.waitUntil(20, new Callable<Integer>() {
@Override
public Integer call() throws Exception {
return list2.size();
} }, TimeUnit.SECONDS, 100);
assertThat(size2).isEqualTo(20);
boolean flag = true;
for(int i =0;i<20;i++){
if(!list1.get(i).equals(list2.get(i))){
flag = false;
break;
}
}
assertThat(flag).isTrue(); ZKDistributedQueue<String> queue = new ZKDistributedQueue(zkClient, rootPath);
assertThat(queue.peek()).isEqualTo(queue.poll());
}
一种基于zookeeper的分布式队列的设计与实现的更多相关文章
- 【连载】redis库存操作,分布式锁的四种实现方式[一]--基于zookeeper实现分布式锁
一.背景 在电商系统中,库存的概念一定是有的,例如配一些商品的库存,做商品秒杀活动等,而由于库存操作频繁且要求原子性操作,所以绝大多数电商系统都用Redis来实现库存的加减,最近公司项目做架构升级,以 ...
- ZooKeeper实现分布式队列Queue
ZooKeeper实现分布式队列Queue 让Hadoop跑在云端系列文章,介绍了如何整合虚拟化和Hadoop,让Hadoop集群跑在VPS虚拟主机上,通过云向用户提供存储和计算的服务. 现在硬件越来 ...
- 基于ZooKeeper的分布式Session实现(转)
1. 认识ZooKeeper ZooKeeper—— “动物园管理员”.动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而不是像走在原始丛林里,心惊胆颤的被 ...
- 基于ZooKeeper的分布式Session实现
1. 认识ZooKeeper ZooKeeper—— “动物园管理员”.动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而不是像走在原始丛林里,心惊胆颤的被 ...
- 基于 Zookeeper 的分布式锁实现
1. 背景 最近在学习 Zookeeper,在刚开始接触 Zookeeper 的时候,完全不知道 Zookeeper 有什么用.且很多资料都是将 Zookeeper 描述成一个“类 Unix/Linu ...
- Java Web学习总结(20)——基于ZooKeeper的分布式session实现
1. 认识ZooKeeper ZooKeeper-- "动物园管理员".动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而不是像走在原始 ...
- 分布式锁(3) ----- 基于zookeeper的分布式锁
分布式锁系列文章 分布式锁(1) ----- 介绍和基于数据库的分布式锁 分布式锁(2) ----- 基于redis的分布式锁 分布式锁(3) ----- 基于zookeeper的分布式锁 代码:ht ...
- 基于Zookeeper的分布式锁(干干干货)
原文地址: https://juejin.im/post/5df883d96fb9a0163514d97f 介绍 为什么使用锁 锁的出现是为了解决资源争用问题,在单进程环境下的资源争夺可以使用 JDK ...
- 基于zookeeper实现分布式配置中心(二)
上一篇(基于zookeeper实现分布式配置中心(一))讲述了zookeeper相关概念和工作原理.接下来根据zookeeper的特性,简单实现一个分布式配置中心. 配置中心的优势 1.各环境配置集中 ...
随机推荐
- list集合如何对里面的元素进行排序
Collections 是集合的公共类,提供各种工具,其中提供了排序方法. Collections.sort(),方法两个参数,1,要排序的集合,2.排序方式 下面是匿名内部类,实现了排序借口,你也可 ...
- java如何实现以数据流的形式下载压缩包到本地?
先不多说,直接贴代码吧,在服务器的E盘下放一个E:/manual.rar的压缩包 package com.cellstrain.icell.controller; import org.springf ...
- java redis基本操作
package com.redis; import java.util.ArrayList;import java.util.Iterator;import java.util.List;import ...
- C++之类和对象的特性
简介:C++并不是一个纯粹的面向对象的语言,而是一种基于过程和面向对象的混合型的语言. 凡是以类对象为基本构成单位的程序称为基于对象的程序,再加上抽象.封装.继承和多态就成为面向对象程序. 1.掌握类 ...
- linux信号量(转载)
本文转载自http://blog.csdn.net/qinxiongxu/article/details/7830537 信号量一.什么是信号量信号量的使用主要是用来保护共享资源,使得资源在一个时刻只 ...
- hdu2041
题目 这道题以前也看到过,但是没有写出来,我刚开始以为用循环遍历一边就可以了,结果我错了,没想到是用的斐波拉契推出来的,用的是递推的思想. 站在楼梯的第n级想一下,前一步是从哪里来的,问题就清楚了. ...
- log4j自动加载原理
java虚拟机加载log4j的类(LogManager.class)后,执行静态代码块,这个类中的静态代码块,会load log4j的配置文件,依次加载log4j.xml,log4j.properti ...
- Android Sqlite 简单SQL语句
--- 创建表 create table student(_id integer primary key autoincrement, name text); --- 查询全部 select _id, ...
- 玩了下STM8单片机
偶然的机会,发现STM8真是又便宜又好用啊,哈哈! 买了一个STM8S103F3的小板子,再加一个ST-Link调试器,总共才35块钱!对于我们这种玩习惯了动辄上千上万的FPGA开发板的人来说,就是白 ...
- IOS下HTML5获取焦点 弹键盘
IOS下sifari和webview默认屏蔽textarea/inputbox获取焦点弹出键盘 苹果移动设备的WebView默认屏蔽textarea/inputbox获取焦点弹出键盘,需要在APP页面 ...