ZooKeeper实现分布式锁

Lock

public interface Lock {public void lock();public void unlock();
}

ZkLock

/*** 使用ZooKeeper实现分布式锁* */
public class ZkLock implements Lock{private CuratorFramework client;private final String zkPath;private Integer count; // 记录获取锁的次数,实现可重入锁private final String subNodePathPrefix;private String lockedPath;private String preNodePath;private String subShortPath;private Thread thread;public ZkLock(CuratorFramework client, String zkPath) throws Exception {this.client = client;this.zkPath = zkPath;subNodePathPrefix = "node-";count = 0;init();}private void init() throws Exception {synchronized (ZkLock.class) {Stat stat;stat = client.checkExists().forPath(zkPath);if(stat == null)client.create().creatingParentsIfNeeded().withMode(CreateMode.PERSISTENT).forPath(zkPath);}}@Overridepublic void lock() {if (thread == Thread.currentThread()) { // 实现可重入锁count += 1;return;}if(lockInternal()){thread = Thread.currentThread();count += 1;}}/*** 上锁函数* */private boolean lockInternal() {try {boolean locked = tryLock(); // 尝试上锁if (locked)return true;// 上锁失败,阻塞等待while (!locked)locked = await();return true;} catch (Exception e) {e.printStackTrace();}return false;}private boolean tryLock() throws Exception {// 为此上锁请求建立临时有序节点lockedPath = client.create().creatingParentsIfNeeded().withMode(CreateMode.PERSISTENT_SEQUENTIAL).forPath(zkPath + "/" + subNodePathPrefix);if (lockedPath == null){throw new Exception();}// 获取锁节点下面所有的上锁请求节点,判断该上锁请求节点的位置,若为首位则获得锁,否则尝试上锁失败subShortPath = getShortPath(lockedPath);List<String> waiters = getWaiters();if(checkIsHeadNode(waiters))return true;int index = Collections.binarySearch(waiters, subShortPath);if (index < 0)throw new Exception();preNodePath = zkPath + "/" + waiters.get(index - 1);return false;}private String getShortPath(String path) {int index = path.lastIndexOf(zkPath + "/");if (index >= 0) {index += zkPath.length() + 1;return index <= path.length() ? path.substring(index) : "";}return null;}private boolean checkIsHeadNode(List<String> waiters) {Collections.sort(waiters);return subShortPath.equals(waiters.get(0));}private List<String> getWaiters() throws Exception {return client.getChildren().forPath(zkPath);}private boolean await() throws Exception {if (preNodePath == null)throw new Exception();CountDownLatch latch = new CountDownLatch(1);client.getData().usingWatcher((Watcher) watchedEvent -> latch.countDown()).forPath(preNodePath);latch.await();return true;}/*** 解锁函数* */@Overridepublic void unlock() {if(!thread.equals(Thread.currentThread()))return;// 检查重入次数count -= 1;if (count > 0)return;try {client.delete().forPath(lockedPath);}catch (Exception e){e.printStackTrace();}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/45474.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot通过企业邮箱发件被Gmail退回的解决方法

这两天给我们开发的Chrome插件&#xff1a;Youtube中文配音 增加了账户注册和登录功能&#xff0c;其中有一步是邮箱验证&#xff0c;所以这边会在Spring Boot后台给用户的邮箱发个验证信息。如何发邮件在之前的文章教程里就有&#xff0c;这里就不说了&#xff0c;着重说说这两…

Lucene教程_编程入门自学教程_菜鸟教程-免费教程分享

教程简介 Lucene是apache软件基金会 jakarta项目组的一个子项目&#xff0c;是一个开放源代码的全文检索引擎工具包&#xff0c;但它不是一个完整的全文检索引擎&#xff0c;而是一个全文检索引擎的架构&#xff0c;提供了完整的查询引擎和索引引擎&#xff0c;部分文本分析引…

通过 kk 创建 k8s 集群和 kubesphere

官方文档&#xff1a;多节点安装 确保从正确的区域下载 KubeKey export KKZONEcn下载 KubeKey curl -sfL https://get-kk.kubesphere.io | VERSIONv3.0.7 sh -为 kk 添加可执行权限&#xff1a; chmod x kk创建 config 文件 KubeSphere 版本&#xff1a;v3.3 支持的 Kuber…

Linux 安全技术和防火墙

目录 1 安全技术 2 防火墙 2.1 防火墙的分类 2.1.1 包过滤防火墙 2.1.2 应用层防火墙 3 Linux 防火墙的基本认识 3.1 iptables & netfilter 3.2 四表五链 4 iptables 4.2 数据包的常见控制类型 4.3 实际操作 4.3.1 加新的防火墙规则 4.3.2 查看规则表 4.3.…

vue3 基于element plus对el-pagination进行二次封装

vue3 基于element plus对el-pagination进行二次封装 1、前言2、在components文件夹中新建pagination.vue文件3、在组件内使用分页 1、前言 在vue3项目中&#xff0c;如果每个列表页都敲一遍分页方法&#xff0c;显然是不合理的&#xff0c;那么&#xff0c;下面我将基于elemen…

企事业数字培训及知识库平台

前言 随着信息化的进一步推进&#xff0c;目前各行各业都在进行数字化转型&#xff0c;本人从事过医疗、政务等系统的研发&#xff0c;和客户深入交流过日常办公中“知识”的重要性&#xff0c;再加上现在倡导的互联互通、数据安全、无纸化办公等概念&#xff0c;所以无论是企业…

打家劫舍 II——力扣213

动规 int robrange(vector<int>& nums, int start, int end){int first=nums[start]

CountDownLatch和CyclicBarrie

前置提要 什么是闭锁对象 闭锁对象&#xff08;Latch Object&#xff09;是一种同步工具&#xff0c;用于控制线程的等待和执行顺序。闭锁对象可以让一个或多个线程等待&#xff0c;直到特定的条件满足后才能继续执行。 在Java中&#xff0c;CountDownLatch就是一种常见的闭锁对…

STC15单片机PM2.5空气质量检测仪

一、系统方案 本设计采用STC15单片机作为主控制器&#xff0c;PM2.5传感器、按键设置&#xff0c;液晶1602显示&#xff0c;蜂鸣器报警。 二、硬件设计 原理图如下&#xff1a; 三、单片机软件设计 1、首先是系统初始化&#xff1a; void lcd_init()//液晶初始化设置 { de…

SQLite数据库实现数据增删改查

当前文章介绍的设计的主要功能是利用 SQLite 数据库实现宠物投喂器上传数据的存储&#xff0c;并且支持数据的增删改查操作。其中&#xff0c;宠物投喂器上传的数据包括投喂间隔时间、水温、剩余重量等参数。 实现功能&#xff1a; 创建 SQLite 数据库表&#xff0c;用于存储宠…

第一讲:BeanFactory和ApplicationContext接口

BeanFactory和ApplicationContext接口 1. 什么是BeanFactory?2. BeanFactory能做什么&#xff1f;3.ApplicationContext对比BeanFactory的额外功能?3.1 MessageSource3.2 ResourcePatternResolver3.3 EnvironmentCapable3.4 ApplicationEventPublisher 4.总结 1. 什么是BeanF…

解决C#报“MSB3088 未能读取状态文件*.csprojAssemblyReference.cache“问题

今天在使用vscode软件C#插件&#xff0c;编译.cs文件时&#xff0c;发现如下warning: 图(1) C#报cache没有更新 出现该warning的原因&#xff1a;当前.cs文件修改了&#xff0c;但是其缓存文件*.csprojAssemblyReference.cache没有更新&#xff0c;需要重新清理一下工程&#x…

【机器学习实战】朴素贝叶斯:过滤垃圾邮件

【机器学习实战】朴素贝叶斯&#xff1a;过滤垃圾邮件 0.收集数据 这里采用的数据集是《机器学习实战》提供的邮件文件&#xff0c;该文件有ham 和 spam 两个文件夹&#xff0c;每个文件夹中各有25条邮件&#xff0c;分别代表着 正常邮件 和 垃圾邮件。 这里需要注意的是需要…

设计模式十六:解释器模式(Interpreter Pattern)

解释器模式是一种行为型设计模式&#xff0c;它用于定义一个语言的文法规则&#xff0c;并且通过解释器来解释执行这些语言中的句子。这种模式通常用于处理一些特定领域的语言&#xff0c;例如编译器、解析器、正则表达式等&#xff0c;解释器模式的核心思想是将一个语言表达式…

【校招VIP】java语言考点之List和扩容

考点介绍&#xff1a; List是最基础的考点&#xff0c;但是很多同学拿不到满分。本专题从两种实现子类的比较&#xff0c;到比较复杂的数组扩容进行分析。 『java语言考点之List和扩容』相关题目及解析内容可点击文章末尾链接查看&#xff01;一、考点题目 1、以下关于集合类…

vue技术学习

vue快速入门 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>vue快速入门</title> </head> <body> <!--老师解读 1. div元素不是必须的&#xff0c;也可以是其它元素&#xff0…

关于flink-sql-connector-phoenix的重写逻辑

目录 重写意义 代码结构 调用链路 POM文件配置 代码解析 一、PhoenixJdbcDynamicTableFactory

在C ++ OpenCV 和 FFTW 中 实现快速去模糊算法

在C ++ OpenCV 和 FFTW 中 实现快速去模糊算法 在图像处理中,模糊是一个常见的问题,它可能由于各种原因(如运动模糊,焦点模糊等)而产生。幸运的是,有一种称为去模糊的技术,可以帮助我们恢复原始的、清晰的图像。在本文中,我们将介绍如何在C++中使用OpenCV和FFTW库实现…

操作系统——操作系统内存管理基础

文章目录 1.内存管理介绍2.常见的几种内存管理机制3.快表和多级页表快表多级页表总结 4.分页机制和分段机制的共同点和区别5.逻辑(虚拟)地址和物理地址6.CPU 寻址了解吗?为什么需要虚拟地址空间? 1.内存管理介绍 操作系统的内存管理主要是做什么&#xff1f; 操作系统的内存…

Apache DolphinScheduler 支持使用 OceanBase 作为元数据库啦!

DolphinScheduler是一个开源的分布式任务调度系统&#xff0c;拥有分布式架构、多任务类型、可视化操作、分布式调度和高可用等特性&#xff0c;适用于大规模分布式任务调度的场景。目前DolphinScheduler支持的元数据库有Mysql、PostgreSQL、H2&#xff0c;如果在业务中需要更好…