一种基于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.各环境配置集中 ...
随机推荐
- Mathtype使用技巧
1. 打开/关闭MathType窗口 Alt+Ctrl+q:插入inline公式 Ctrl+S:更新公式到Word相应位置 Alt+F4:保存并关闭MathType窗口,返回Word. 2. 公式 ...
- 检测空值,以及会不会出现mapping类型不一致的问题
/// <summary> /// 检测空值,以及会不会出现mapping类型不一致的问题 /// </summary> /// <typeparam name=&quo ...
- spring mvc , spring data jpa 知识点
<mvc:view-controller path="/login" view-name="login.jsp"/> <!-- 会直接跳到/W ...
- css长度
在CSS样式表中,长度单位分两种: 相对长度单位,如px, em等绝对长度单位,如pt,mm等 CSS相对长度单位(relative length unit) CSS相对长度单位中的相对二字,表明了其 ...
- SQLServerException:将截断字符串或二进制数据的解决方法
SQLServerException:将截断字符串或二进制数据的解决方法: 最近使用JPA进行保存对象到数据库中怎么也添加不进去,始终报错 主要原因就是你增加的数据字段长度超过数据库中字段所定义长度, ...
- 2018.07.10NOIP模拟 Knapsack(单调队列优化dp)
Knapsack 题目背景 SOURCE:NOIP2016-RZZ-4 T2 题目描述 有 n 个物品,第 i 个物品的重量为 ai . 设 f(i,j,k,l,m) 为满足以下约束的物品集合数量: ...
- 34.More-healthyful Menu Options for Kids 给孩子提供的更健康菜式选择
34.More-healthyful Menu Options for Kids 给孩子提供的更健康菜式选择 ① French fries,washed down with a pint of sod ...
- HDU 2546 饭卡 (01背包问题)
题意:中文的吧,飘过~ 析:学过DP的都应该感觉到是动态规划吧,就是一个01背包问题,不同的是,这个题又加入一些新的条件,就是不满5元不能消费,过了5元即使超了也行(这个学校真不错,都可以预支),最后 ...
- 全面理解iOS开发中的Scroll View[转]
from:http://mobile.51cto.com/hot-430409.htm 可能你很难相信,UIScrollView和一个标准的UIView差异并不大,scroll view确实会多一些方 ...
- (KMP 扩展)Clairewd’s message -- hdu -- 4300
http://acm.hdu.edu.cn/showproblem.php?pid=4300 Clairewd’s message Time Limit: 2000/1000 MS (Java/Oth ...