一、使用场景

  在分布式应用,往往存在多个进程提供同一服务。这些进程有可能在相同的机器上,也有可能分布在不同的机器上。 如果这些进程共享了一些资源,可能就需要分布式锁来锁定对这些资源的访问。

二、实现分布式锁结构图

三、代码实现

 package com.xbq.zookeeper.javaApi;

 import java.util.Collections;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit; import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.WatchedEvent;
import org.apache.zookeeper.Watcher;
import org.apache.zookeeper.ZooDefs.Ids;
import org.apache.zookeeper.ZooKeeper;
import org.apache.zookeeper.data.Stat; /**
* 使用Zookeeper原生API实现分布式锁
* @author xbq
*/
public class ZookeeperLock implements Watcher{ // 声明zk对象
private ZooKeeper zk = null;
// 此demo使用的集群,所以有多个ip和端口
private static String CONNECT_SERVER = "192.168.242.130:2181,192.168.242.130:2182,192.168.242.130:2183";
// session过期时间
private static int SESSION_TIMEOUT = 3000;
// 根节点
private String root = "/locks";
// 当前等待的节点
private String waitNode;
// 等待的时间
private int waitTime;
// 锁节点
private String myzkNode;
// 计数器
private CountDownLatch latch; /**
* 构造函数 初始化
*/
public ZookeeperLock(){
try {
zk = new ZooKeeper(CONNECT_SERVER, SESSION_TIMEOUT, this);
// 判断是否存在根节点,不需要监听根节点
Stat stat = zk.exists(root, false);
// 为空,说明不存在
if(stat == null){
// 添加一个永久节点,即添加一个根节点
zk.create(root, new byte[0], Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
}
} catch (Exception e) {
e.printStackTrace();
}
} /**
* 尝试获取锁
* @return
*/
private boolean tryLock(){
String splitStr = "lock_"; // 格式 lock_000000001
try {
// 创建一个临时有序节点,并赋值给 myzkNode
myzkNode = zk.create(root + "/" + splitStr, new byte[0], Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
// 获取根节点下的所有子节点
List<String> children = zk.getChildren(root, false);
// 对子节点 排序
Collections.sort(children);
// 如果刚刚创建的节点 等于 获取最小的一个子节点,则说明 获取到锁
if(myzkNode.equals(root + "/" + children.get(0))){
return true;
}
// 如果刚刚创建的节点 不等于 获取到的最小的一个子节点,则 监控比自己小的一个节点
// 获取刚刚建立的子节点(不包含根节点的子节点)
String childNode = myzkNode.substring(myzkNode.lastIndexOf("/") + 1);
// 获取比自己小的节点
waitNode = children.get(Collections.binarySearch(children, childNode) - 1);
} catch (KeeperException e) {
e.printStackTrace();
} catch (InterruptedException e) {
e.printStackTrace();
}
return false;
} /**
* 等待锁释放
* @param waitNode
* @param waidTime
* @return
* @throws KeeperException
* @throws InterruptedException
*/
private boolean waitLock(String waitNode, int waidTime) throws KeeperException, InterruptedException{
// 判断比自己小的节点是否存在,并添加监听
Stat stat = zk.exists(root + "/" + waitNode, true);
// 如果存在 比自己小的节点
if(stat != null){
this.latch = new CountDownLatch(1);
this.latch.await(waidTime, TimeUnit.MILLISECONDS);
this.latch = null;
}
return true;
} /**
* 获取锁
*/
public void lock(){
// 如果尝试获取锁成功
if(tryLock()){
// 获取 成功
System.out.println("Thread" + Thread.currentThread().getName() + " -- hold lock!");
return;
}
// 等待并获取锁
try {
waitLock(waitNode, waitTime);
} catch (KeeperException e) {
e.printStackTrace();
} catch (InterruptedException e) {
e.printStackTrace();
}
} /**
* 解锁
*/
public void unLock(){
try {
zk.delete(myzkNode, -1);
zk.close();
System.out.println("Thread" + Thread.currentThread().getName() +" unlock success! 锁节点:" + myzkNode);
} catch (InterruptedException e) {
e.printStackTrace();
} catch (KeeperException e) {
e.printStackTrace();
}
} /**
* 删除的时候 触发事件
*/
@Override
public void process(WatchedEvent event) {
// 如果计数器不为空的话,释放计数器锁
if(this.latch != null){
this.latch.countDown();
}
} /**
* 测试
* @param args
*/
public static void main(String[] args) {
// 定义线程池
ExecutorService service = Executors.newCachedThreadPool();
// 只能10个线程同时运行,即模拟并发数为10
final Semaphore semaphore = new Semaphore(10);
// 10个客户端连接
for(int i=0;i<10;i++){
Runnable runnable = new Runnable() {
@Override
public void run() {
try {
semaphore.acquire();
ZookeeperLock zkLock = new ZookeeperLock();
zkLock.lock();
// 业务操作代码
Thread.sleep(3000);
zkLock.unLock();
semaphore.release();
} catch (Exception e) {
e.printStackTrace();
}
}
};
service.execute(runnable);
}
service.shutdown();
}
}

源码下载:https://gitee.com/xbq168/DistributedLockByRedis

ZooKeeper(七)-- ZK原生API实现分布式锁的更多相关文章

  1. 使用ZooKeeper实现Java跨JVM的分布式锁(优化构思)

    一.使用ZooKeeper实现Java跨JVM的分布式锁 二.使用ZooKeeper实现Java跨JVM的分布式锁(优化构思) 三.使用ZooKeeper实现Java跨JVM的分布式锁(读写锁) 说明 ...

  2. 使用ZooKeeper实现Java跨JVM的分布式锁(读写锁)

    一.使用ZooKeeper实现Java跨JVM的分布式锁 二.使用ZooKeeper实现Java跨JVM的分布式锁(优化构思) 三.使用ZooKeeper实现Java跨JVM的分布式锁(读写锁) 读写 ...

  3. 使用ZooKeeper实现Java跨JVM的分布式锁

    一.使用ZooKeeper实现Java跨JVM的分布式锁 二.使用ZooKeeper实现Java跨JVM的分布式锁(优化构思) 三.使用ZooKeeper实现Java跨JVM的分布式锁(读写锁) 说明 ...

  4. curator框架的使用以及实现分布式锁等应用与zkclient操作zookeeper,简化复杂原生API

    打开zookeeper集群 先体会一下原生API有多麻烦(可略过): //地址 static final String ADDR = "192.168.171.128:2181,192.16 ...

  5. Zookeeper系列2 原生API 以及核心特性watcher

    原生API 增删改查询 public class ZkBaseTest { static final String CONNECT_ADDR = "192.168.0.120"; ...

  6. ZooKeeper(八)-- Curator实现分布式锁

    1.pom.xml <dependencies> <dependency> <groupId>junit</groupId> <artifactI ...

  7. Zookeeper系列三:Zookeeper客户端的使用(Zookeeper原生API如何进行调用、ZKClient、Curator)和Zookeeper会话

    一.Zookeeper原生API如何进行调用 准备工作: 首先在新建一个maven项目ZK-Demo,然后在pom.xml里面引入zk的依赖 <dependency> <groupI ...

  8. ZooKeeper分布式锁的实现原理

    七张图彻底讲清楚ZooKeeper分布式锁的实现原理[石杉的架构笔记] 文章转载自:https://juejin.im/post/5c01532ef265da61362232ed#comment(写的 ...

  9. ZooKeeper 分布式锁

    在Redis分布式锁一文中, 作者介绍了如何使用Redis开发分布式锁. Redis分布式锁具有轻量高吞吐量的特点,但是一致性保证较弱.我们可以使用Zookeeper开发分布式锁,来满足对高一致性的要 ...

随机推荐

  1. Java EE各种documentation

    总之通用的法则,google:"XxxName documentation" 例如,如果要查看JSTL的reference manual,google:jstl doc即可,ora ...

  2. js中Math.round、parseInt、Math.floor和Math.ceil小数取整小结【转】

    [摘要:之前常常正在代码中看到Math.round.parseInt.Math.floor战Math.ceil那四个函数,固然晓得效果皆能够返回一个整数,然则对他们四者的差别照样没有太清晰,本日便做一 ...

  3. 一款简洁的纯css3代码实现的动画导航

    之前为大家介绍了好多导航菜单,今天给大家分享一款简洁的纯css3代码实现的动画导航.鼠标经过的时候以背景色以菱形渐变为长方形.效果图如下: 在线预览   源码下载 实现的代码. html代码: < ...

  4. Curved UI - VR Ready Solution To Bend Warp Your Canvas 1.7,1.8,2.2,2.3 四种版本压缩包(Unity UGUI曲面插件),可以兼容VRTK

    Curved UI - VR Ready Solution To Bend Warp Your Canvas 1.7,1.8,2.2,2.3 四种版本压缩包(Unity UGUI曲面插件) 可以兼容V ...

  5. java——关于异常处理机制的简单原理和应用

    异常处理机制的简单原理和应用 一.Execption可以分为java标准定义的异常和程序员自定义异常2种 (1)一种是当程序违反了java语规则的时候,JAVA虚拟机就会将发生的错误表示为一个异常.这 ...

  6. 启动hive出错,提示没有权限

    报错信息如下: which: no hbase in (/usr/local/sqoop-1.4.6.bin__hadoop-2.0.4-alpha/bin:/usr/local/hive/bin:/ ...

  7. Nginx实战系列之功能篇----后端节点健康检查(转)

    公司前一段对业务线上的nginx做了整理,重点就是对nginx上负载均衡器的后端节点做健康检查.目前,nginx对后端节点健康检查的方式主要有3种,这里列出:   1.ngx_http_proxy_m ...

  8. Hive中order by,sort by,distribute by,cluster by的区别

    一:order by order by会对输入做全局排序,因此只有一个Reducer(多个Reducer无法保证全局有序),然而只有一个Reducer,会导致当输入规模较大时,消耗较长的计算时间.关于 ...

  9. 字符数组在C++、C#等语言中的操作

    1.C++中操作数组 #include <iostream> using namespace std; int length(char []); void output_frequency ...

  10. Nutch系列1:简介

    由Java实现的,开放源代码(open-source)的web搜索引擎. Nutch 致力于让每个人都能很容易, 同时花费很少就可以配置世界一流的Web搜索引擎. 为了完成这一宏伟的目标, Nutch ...