zk实现分布式锁
public interface lock {
void getLock();
void unLock();
}
public abstract class ZkAbstractLock implements lock{
//zk连接地址
private static final String CONNECTSTRING = "127.0.0.1:2181";
//创建zk连接
protected ZkClient zkClient = new ZkClient(CONNECTSTRING);
//PATH
protected static final String PATH = "/lock";
public abstract boolean tryLock();
public abstract void waitLock();
public void getLock() {
if (tryLock()) {
System.out.println("###成功获取锁###");
} else {
waitLock();
getLock();
}
}
public void unLock() {
if(zkClient!=null){
zkClient.close();
System.out.println("###释放所资源###");
}
}
}
**
* @Auther:
* @Date: 星期一:2019/4/29 15
* @Description: zk基于异常的阻塞式锁
*/
public class ZKExceptionLock extends ZkAbstractLock {
private String lockPath;
private CountDownLatch countDownLatch = new CountDownLatch(1); public ZKExceptionLock(String lockPath) {
this.lockPath = PATH + "/" + lockPath;
}
public boolean tryLock() {
try {
zkClient.createEphemeral(lockPath);
return true;
}catch (Exception e){
return false;
}
} public void waitLock() {
zkClient.subscribeDataChanges(lockPath, new IZkDataListener() {
public void handleDataChange(String s, Object o) throws Exception {
System.out.println("---handleDataChange---");
} public void handleDataDeleted(String s) throws Exception {
countDownLatch.countDown();
}
}); try {
countDownLatch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
@Override
public void unLock(){
zkClient.delete(lockPath);
}
}
/**
* @Auther:
* @Date: 星期一:2019/4/29 12
* @Description: zk基于监听的阻塞式锁
*/
public class ZkListenerLock extends ZkAbstractLock {
private static final CountDownLatch LATCH = new CountDownLatch(1);
private String lockPath;
private String currentPath;
private String beforePath; public ZkListenerLock(String lockPath) {
if (!zkClient.exists(PATH)) {
zkClient.createPersistent(PATH);
}
this.lockPath = PATH+"/"+lockPath;
if (!zkClient.exists(this.lockPath)) {
System.out.println("-----------");
zkClient.createPersistent(this.lockPath);
}
} public boolean tryLock() {
if (null == currentPath || currentPath.length()<=0 ) {
currentPath = zkClient.createEphemeralSequential(lockPath + "/", "lock");
}
List<String> children = zkClient.getChildren(lockPath);//0000000007
Collections.sort(children);
System.out.println(children);
if (currentPath.equals(lockPath + "/"+children.get(0))) {
return true;
}
String key = currentPath.substring(lockPath.length() + 1);
int location = Collections.binarySearch(children, key);
this.beforePath = lockPath + "/" + children.get(location - 1);
return false;
} public void waitLock() {
IZkDataListener listener = new IZkDataListener() {
public void handleDataChange(String s, Object o) throws Exception {
System.out.println("--handleDataChange--");
}
public void handleDataDeleted(String s) throws Exception {
LATCH.countDown();
}
};
zkClient.subscribeDataChanges(this.beforePath,listener);
if (!zkClient.exists(this.beforePath)) {
return;
}
try {
LATCH.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
zkClient.unsubscribeDataChanges(this.beforePath,listener);
} @Override
public void unLock() {
zkClient.delete(currentPath);
}
}
public class TestLock {
public static void main(String[] args) {
int i = 0;
while (i < 10) {
People people = new People();
people.start();
i++;
}
System.out.println(People.totalMoney);
}
static class People extends Thread{
private ZkListenerLock lock = new ZkListenerLock("order");
private ZKExceptionLock lock2 = new ZKExceptionLock("order2");
public static int totalMoney = 50;
public void run(){
try {
//获取锁资源
lock2.getLock();
totalMoney--;
} catch (Exception e) {
e.printStackTrace();
} finally {
//释放锁资源
lock2.unLock();
}
}
}
}
基于监听的zk锁:L值就是建立有序节点后返回的序列号

基于异常的zk锁:

pom.xml:
<dependencies>
<dependency>
<groupId>com.101tec</groupId>
<artifactId>zkclient</artifactId>
<version>0.10</version>
</dependency>
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-lang3</artifactId>
<version>3.3.2</version>
</dependency>
</dependencies>
zk实现分布式锁的更多相关文章
- zookeeper笔记之基于zk实现分布式锁
一.分布式锁概述 Java中基于AQS框架提供了一系列的锁,但是当需要在集群中的多台机器上互斥执行一段代码或使用资源时Java提供的这种单机锁就没了用武之地,此时需要使用分布式锁协调它们.分布式锁有很 ...
- 基于ZK的分布式锁实现
import java.util.concurrent.TimeUnit; import org.apache.curator.framework.CuratorFramework; import o ...
- 2020-06-18:ZK的分布式锁怎么实现?
福哥答案2020-06-18: Zk分布式锁有两种实现方式一种比较简单,应对并发量不是很大的情况.获得锁:创建一个临时节点,比如/lock,如果成功获得锁,如果失败没获得锁,返回false释放锁:删除 ...
- 【zookeeper】Apache curator的使用及zk分布式锁实现
上篇,本篇主要讲Apache开源的curator的使用,有了curator,利用Java对zookeeper的操作变得极度便捷. 其实在学之前我也有个疑虑,我为啥要学curator,撇开涨薪这些外在的 ...
- 分布式锁-基于ZK和Redis实现
一.基于zookeeper实现分布式锁 1.1 Zookeeper的常用接口 package register; import java.util.List; import java.util.con ...
- 面试必问:分布式锁实现之zk(Zookeeper)
点赞再看,养成习惯,微信搜索[三太子敖丙]关注这个互联网苟且偷生的工具人. 本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试完整考点.资料以及我的 ...
- 【分布式锁的演化】终章!手撸ZK分布式锁!
前言 这应该是分布式锁演化的最后一个章节了,相信很多小伙伴们看完这个章节之后在应对高并发的情况下,如何保证线程安全心里肯定也会有谱了.在实际的项目中也可以参考一下老猫的github上的例子,当然代码没 ...
- 一般实现分布式锁都有哪些方式?使用redis如何设计分布式锁?使用zk来设计分布式锁可以吗?这两种分布式锁的实现方式哪种效率比较高?
#(1)redis分布式锁 官方叫做RedLock算法,是redis官方支持的分布式锁算法. 这个分布式锁有3个重要的考量点,互斥(只能有一个客户端获取锁),不能死锁,容错(大部分redis节点创建了 ...
- zk系列三:zookeeper实战之分布式锁实现
一.分布式锁的通用实现思路 分布式锁的概念以及常规解决方案可以参考之前的博客:聊聊分布式锁的解决方案:今天我们先分析下分布式锁的实现思路: 首先,需要保证唯一性,即某一时点只能有一个线程访问某一资源: ...
随机推荐
- python学习day4 数据类型 if语句
1.变量的内存管理 cpython解释器垃圾回收机制 什么是垃圾,当一个值身上没有绑定变量名时,(该值的引用计数=0时)就是一个垃圾 age=18 #18的引用计数=1 x=age #18的引用计数 ...
- css设计技巧
如果设置了样式发现没有效果,可以把需要调的元素或者父元素等设置一下背景,然后看看哪个父元素或子元素有样式控制,可以清除一下.
- 侯捷STL课程及源码剖析学习2: allocator
以STL 的运用角度而言,空间配置器是最不需要介绍的东西,它总是隐藏在一切组件(更具体地说是指容器,container)的背后,默默工作默默付出. 一.分配器测试 测试代码 #include < ...
- 非常棒的轨迹插件Better Trails v1.4.6
点击下载
- MYSQL之水平分区----MySQL partition分区I(5.1)
一. 分区的概念 二. 为什么使用分区?(优点) 三. 分区类型 四. 子分区 五. 对分区进行修改(增加.删除.分解.合并) 六 ...
- vue中使用vue-router切换页面时滚动条自动滚动到顶部的方法
原文:http://www.jb51.net/article/129270.htm main.js入口文件配合vue-router写这个 router.afterEach((to,from,next) ...
- [剑指Offer]54-二叉搜索树的第k个节点
题目描述 给定一棵二叉搜索树,找出其中的第k小的结点,返回指向该节点的指针. 思路 中序遍历即可. 注意特判!报段错误数组越界这里就要考虑是少特判的问题. 法一:借助vector 法二(better) ...
- 不常见的javascript调试技巧
原文链接:https://segmentfault.com/a/1190000011857058 有时, 有一组复杂的对象要查看.可以通过console.log查看并滚动浏览,亦或者使用console ...
- 关于vuex状态管理模式架构
一. 什么是vuex 集中存储管理所有组件的状态 并以相应的规则保证以一种可预测的方式发生变化. 例子: 实现加减 <p>{{count}} <button @click=" ...
- 线特征---LSD算法(二)
上一节介绍了一些资源和实验结果,这节主要是介绍LSD算法理论. 直线段检测算法---LSD:a Line Segment Detector LSD的核心是像素合并于误差控制.利用合并像素来检测直线段并 ...