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实现分布式锁的更多相关文章

  1. zookeeper笔记之基于zk实现分布式锁

    一.分布式锁概述 Java中基于AQS框架提供了一系列的锁,但是当需要在集群中的多台机器上互斥执行一段代码或使用资源时Java提供的这种单机锁就没了用武之地,此时需要使用分布式锁协调它们.分布式锁有很 ...

  2. 基于ZK的分布式锁实现

    import java.util.concurrent.TimeUnit; import org.apache.curator.framework.CuratorFramework; import o ...

  3. 2020-06-18:ZK的分布式锁怎么实现?

    福哥答案2020-06-18: Zk分布式锁有两种实现方式一种比较简单,应对并发量不是很大的情况.获得锁:创建一个临时节点,比如/lock,如果成功获得锁,如果失败没获得锁,返回false释放锁:删除 ...

  4. 【zookeeper】Apache curator的使用及zk分布式锁实现

    上篇,本篇主要讲Apache开源的curator的使用,有了curator,利用Java对zookeeper的操作变得极度便捷. 其实在学之前我也有个疑虑,我为啥要学curator,撇开涨薪这些外在的 ...

  5. 分布式锁-基于ZK和Redis实现

    一.基于zookeeper实现分布式锁 1.1 Zookeeper的常用接口 package register; import java.util.List; import java.util.con ...

  6. 面试必问:分布式锁实现之zk(Zookeeper)

    点赞再看,养成习惯,微信搜索[三太子敖丙]关注这个互联网苟且偷生的工具人. 本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试完整考点.资料以及我的 ...

  7. 【分布式锁的演化】终章!手撸ZK分布式锁!

    前言 这应该是分布式锁演化的最后一个章节了,相信很多小伙伴们看完这个章节之后在应对高并发的情况下,如何保证线程安全心里肯定也会有谱了.在实际的项目中也可以参考一下老猫的github上的例子,当然代码没 ...

  8. 一般实现分布式锁都有哪些方式?使用redis如何设计分布式锁?使用zk来设计分布式锁可以吗?这两种分布式锁的实现方式哪种效率比较高?

    #(1)redis分布式锁 官方叫做RedLock算法,是redis官方支持的分布式锁算法. 这个分布式锁有3个重要的考量点,互斥(只能有一个客户端获取锁),不能死锁,容错(大部分redis节点创建了 ...

  9. zk系列三:zookeeper实战之分布式锁实现

    一.分布式锁的通用实现思路 分布式锁的概念以及常规解决方案可以参考之前的博客:聊聊分布式锁的解决方案:今天我们先分析下分布式锁的实现思路: 首先,需要保证唯一性,即某一时点只能有一个线程访问某一资源: ...

随机推荐

  1. JMeter学习(八)JDBC测试计划-连接Oracle(转载)

    转载自 http://www.cnblogs.com/yangxia-test 一.测试环境准备   Oracle:10g  JDBC驱动:classes12.jar oracle安装目录下(orac ...

  2. ORA-01555 snapshot too old

    假设有一张6000万行数据的testdb表,预计testdb全表扫描1次需要2个小时,参考过程如下: 1.在1点钟,用户A发出了select * from testdb;此时不管将来testdb怎么变 ...

  3. cloudera cdh5.13.0 vmware 快速安装

    1. 从官网上载VMWARE VM快速安装包 https://www.cloudera.com/downloads/quickstart_vms/5-12.html 2. 下载后的安装包,解压之后得到 ...

  4. 第二章 向量(c)无序向量

  5. webpack 安装,打包使用

      Webpack 本身只能处理 JavaScript 模块,如果要处理其他类型的文件,就需要使用 loader 进行转换. 全局安装webpack 打开文件夹amd输入指令  npm i webpa ...

  6. ubuntu下安装nginx1.11.10

    (本页仅作为个人笔记参考) 为openssl,zlib,pcre配置编译 wget http://om88fxbu9.bkt.clouddn.com/package/nginx/nginx-1.11. ...

  7. utils.js和vue-loader.conf.js

    var utils = require('./utils')var config = require('../config')var isProduction = process.env.NODE_E ...

  8. 因为要生成的折线数量是不定的 ,所以需要echarts 动态的为option中的Series添加数据

    series:function(){ var serie=[]; var aa = zhonglei[0].split(","); for( var i=0;i < aa.l ...

  9. [z]一分钟教你知道乐观锁和悲观锁的区别

    悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁.传统的关系型数据 ...

  10. mysql主备配置

    目录 mysql主备2 一.master配置:2 1. 修改配置文件 2 2. 登录添加账号并赋权限 2 3. 查看master信息 2 二.slave配置:2 1. 修改配置文件 2 2. 重启登录 ...