分布式锁系列文章

分布式锁(1) ----- 介绍和基于数据库的分布式锁

分布式锁(2) ----- 基于redis的分布式锁

分布式锁(3) ----- 基于zookeeper的分布式锁

代码:https://github.com/shuo123/distributeLock

思路

利用zookeeper的临时有序节点和watch实现,思路如下:

  1. 获取锁的请求在锁的根目录下调用create()创建临时有序节点
  2. 在锁的根目录下调用getChildren()获取所有子节点
  3. 判断步骤1创建的临时有序节点是否是所有子节点中最小的一个,是则获取锁,结束获取
  4. 否则对排在该节点的前一个节点调用exists()判断节点是否存在,并设置watch监听节点的删除事件
  5. 如果步骤4返回节点不存在转到步骤2,否则等待节点删除再转到步骤2

注意:

  1. 创建临时节点,是在请求锁的客户端挂了,节点会自动删除
  2. 创建有序节点,是为了防止群体效应,只需监听前一个节点的删除事件,不用监听所有节点的删除事件
  3. 该思路实现的是公平锁

实现

java实现

public boolean tryLock(long waitTime){
try {
//1.创建临时有序节点
myZNode = zk.create(root + "/" + lockName + "-", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
return waitForLock(waitTime);
} catch (KeeperException e) {
e.printStackTrace();
} catch (InterruptedException e) {
e.printStackTrace();
}
return false;
} private boolean waitForLock(long watiTime) {
long start = System.currentTimeMillis();
try {
while (System.currentTimeMillis() - start < watiTime) {
//2.获取子节点
List<String> children = zk.getChildren(root, false);
List<String> lockNodes = new ArrayList<>();
for (String s : children) {
if (s.startsWith(lockName)) {
lockNodes.add(s);
}
}
Collections.sort(lockNodes);
//3.判断是否最小节点
if (myZNode.equals(root + "/" + lockNodes.get(0))) {
return true;
}
//4.监听前一个节点删除事件
String seq = myZNode.substring(myZNode.lastIndexOf('/') + 1);
String waitNode = lockNodes.get(Collections.binarySearch(lockNodes, seq) - 1);
CountDownLatch latch = new CountDownLatch(1);
Stat stat = zk.exists(root + "/" + waitNode, watchedEvent -> {
if (watchedEvent.getType() == Watcher.Event.EventType.NodeDeleted) {
latch.countDown();
}
});
if (stat != null) {
latch.await(watiTime - System.currentTimeMillis() + start, TimeUnit.MILLISECONDS);
}
}
}catch (Exception e){
deleteNode();
e.printStackTrace();
}
deleteNode();
return false;
} public boolean unlock() {
return deleteNode();
} private boolean deleteNode(){
try {
zk.delete(myZNode, -1);
return true;
} catch (InterruptedException e) {
e.printStackTrace();
} catch (KeeperException e) {
e.printStackTrace();
}
return false;
}

测试代码

private static class ZoookeeperLockTest implements Runnable {

    private CyclicBarrier barrier;

    ZoookeeperLockTest(CyclicBarrier barrier) {
this.barrier = barrier;
} @Override
public void run() {
try {
CountDownLatch latch = new CountDownLatch(1);
ZooKeeper zk = new ZooKeeper("127.0.0.1:2181,127.0.0.1:2182,127.0.0.1:2183", 5000, WatchedEvent -> {
latch.countDown();
});
latch.await();
try {
DistributeLock lock = new ZookeeperLock(zk, "lock");
barrier.await();
lock.tryLock(Integer.MAX_VALUE);
try {
System.out.println(Thread.currentThread().getName() + "get lock");
Thread.sleep(100);
System.out.println(Thread.currentThread().getName() + "get unlock");
} finally {
lock.unlock();
}
} finally {
zk.close();
}
} catch (Exception e) {
e.printStackTrace();
}
}
}

缺点与解决方案

没有实现重入。可以使用threadlocal保存重入次数,每次解锁-1,到0时才删除节点

curator

curator是apache的顶级开源项目,实现了分布式锁及队列、选举等多种高级功能。同时优化了zookeeper原生api的很多问题,如支持自动连接,递归创建删除节点等等。

private static class CuratorLockTest implements Runnable {

    private CyclicBarrier barrier;

    CuratorLockTest(CyclicBarrier barrier) {
this.barrier = barrier;
} @Override
public void run() {
RetryPolicy retryPolicy = new ExponentialBackoffRetry(1000, 3);
CuratorFramework client = CuratorFrameworkFactory.builder()
.connectString("127.0.0.1:2181,127.0.0.1:2182,127.0.0.1:2183")
.sessionTimeoutMs(5000)
.retryPolicy(retryPolicy)
.build();
client.start();
InterProcessLock lock = new InterProcessMutex(client, "/locks/curator-lock");
try{
lock.acquire();
try {
System.out.println(Thread.currentThread().getName() + "get lock");
Thread.sleep(100);
System.out.println(Thread.currentThread().getName() + "get unlock");
}finally {
lock.release();
}
} catch (Exception e) {
e.printStackTrace();
}finally {
client.close();
}
}
}

参考资料

http://zookeeper.apache.org/doc/r3.4.13/recipes.html#sc_recipes_Locks

https://www.cnblogs.com/seesun2012/p/9214653.html

分布式锁(3) ----- 基于zookeeper的分布式锁的更多相关文章

  1. 基于Zookeeper的分布式锁(干干干货)

    原文地址: https://juejin.im/post/5df883d96fb9a0163514d97f 介绍 为什么使用锁 锁的出现是为了解决资源争用问题,在单进程环境下的资源争夺可以使用 JDK ...

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

    1. 背景 最近在学习 Zookeeper,在刚开始接触 Zookeeper 的时候,完全不知道 Zookeeper 有什么用.且很多资料都是将 Zookeeper 描述成一个“类 Unix/Linu ...

  3. 【连载】redis库存操作,分布式锁的四种实现方式[一]--基于zookeeper实现分布式锁

    一.背景 在电商系统中,库存的概念一定是有的,例如配一些商品的库存,做商品秒杀活动等,而由于库存操作频繁且要求原子性操作,所以绝大多数电商系统都用Redis来实现库存的加减,最近公司项目做架构升级,以 ...

  4. 基于ZooKeeper的分布式Session实现(转)

    1.   认识ZooKeeper ZooKeeper—— “动物园管理员”.动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而不是像走在原始丛林里,心惊胆颤的被 ...

  5. 基于ZooKeeper的分布式Session实现

    1.   认识ZooKeeper ZooKeeper—— “动物园管理员”.动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而不是像走在原始丛林里,心惊胆颤的被 ...

  6. Java Web学习总结(20)——基于ZooKeeper的分布式session实现

    1.   认识ZooKeeper ZooKeeper-- "动物园管理员".动物园里当然有好多的动物,游客可以根据动物园提供的向导图到不同的场馆观赏各种类型的动物,而不是像走在原始 ...

  7. ShardingJdbc基于Zookeeper实现分布式治理

    随着数据规模的不断膨胀,使用多节点集群的分布式方式逐渐成为趋势.在这种情况下,如何高效.自动化管理集群节点,实现不同节点的协同工作,配置一致性,状态一致性,高可用性,可观测性等,就成为一个重要的挑战. ...

  8. 基于zookeeper实现分布式配置中心(二)

    上一篇(基于zookeeper实现分布式配置中心(一))讲述了zookeeper相关概念和工作原理.接下来根据zookeeper的特性,简单实现一个分布式配置中心. 配置中心的优势 1.各环境配置集中 ...

  9. 基于ZooKeeper的分布式锁和队列

    在分布式系统中,往往需要一些分布式同步原语来做一些协同工作,上一篇文章介绍了Zookeeper的基本原理,本文介绍下基于Zookeeper的Lock和Queue的实现,主要代码都来自Zookeeper ...

随机推荐

  1. 操作系统识别-python、nmap

    识别操作系统主要是用于操作系统漏洞的利用.不管是windows还是linux系统,在安装完毕后都会默认启动一些服务,开启一些端口. 识别目标主机的系统最简单的方法就是发送ping包,windows起始 ...

  2. 微信小程序上传Word文档、PDF、图片等文件

    <view class="main" style="border:none"> <view class="title"&g ...

  3. 实时web应用方案——SignalR(.net core)

    何为实时 先从理论上解释一下两者的区别. 大多数传统的web应用是这样的:客户端发起http请求到服务端,服务端返回对应的结果.像这样: 也就是说,传统的web应用都是客户端主动发起请求到服务端. 那 ...

  4. javamelody对Java Application进行监控

    前面写过对于webapp,用javamelody来监控.分析性能是挺方便的:那要对普通的java应用进行监控,只需要在应用上启动一个嵌入式web容器就可以了. javamelody里面的war包就用了 ...

  5. 【搬砖】/etc/passwd 文件结构

    文件的内容: 每个Linux都会有root账号,以此来说明,从内容上看,每行都使用[:]分隔(注意:每行前面的数字是行号,不是内容),所以一共有七部分内容,分别是: 1. 账号名称:root,用来对应 ...

  6. 自定义PHPstorm快捷键

    这篇随笔介绍一下PHPstorm自定义快捷键的步骤: 1.点击主菜单 File 下的 setting : 2.在弹出框中点击 Keymap : 3.之后会出现如下图所示的界面(图中所有的菜单都折叠了) ...

  7. 常用API - Arrays、Math、Object

    Arrays类 概述 此类包含用来操作数组(比如排序和搜索)的各种方法.此类还包含一个允许将数组作为列表来查看的静态工厂. 除非特别注明,否则如果指定数组引用为 null,则此类中的方法都会抛出 Nu ...

  8. HTML5(四)Drag and Drop

    HTML5 拖放(Drag 和 Drop) 拖放 拖放是一种常见的特性,即抓取对象以后拖到另一个位置. 在 HTML5 中,拖放是标准的一部分,任何元素都能够拖放. 设置元素为可拖放 首先,为了使元素 ...

  9. bootstrap悬停下拉菜单显示

    使用Bootstrap导航条组件时,如果你的导航条带有下拉菜单,那么这个带下拉菜单的导航在点击时只会浮出下拉菜单,它本身的href属性会失效,也就是失去了超链接功能,这并不是我想要的,我希望导航条的链 ...

  10. adb devices 不能连接设备 could not install *smartsocket* listener

    cmd以管理员身份运行命令adb devices  或adb reverse tcp:8081 tcp:8081,无法连接设备,出现上图信息. 输入命令:adb kill-server 再输入:adb ...