分布式锁 ----zookeeper实践 (排它锁)
排它锁概念:
Exclusive Locks,被称为X锁,写锁,独占锁.如果事物T1对数据对象O1加上了排它锁,那么在整个加锁期间,只允许事务T1对O1进行读写操作,其他事务必须等到T1释放锁后才能进行操作.在单机环境中,JDK提供了synchronized关键字和ReentrantLock
重用锁来提供排它锁的功能.
zookeeper实现排它锁原理:
在需要获取排它锁时,所有的客户端都会调用create方法在固定路径下创建节点,并发环境下,只有一个客户端可以创建成功,相当于获取了锁,当该客户端完成事务后,删除该节点.对于没有获取成功的其他节点,则在该路径上设置监听,如果该路径子节点删除事件触发,继续尝试获取锁(create).
具体实现:
定义锁的接口:(我想实现的几个方法)
void lock();
boolean isLocked();
boolean tryLock();
boolean tryLock(long timeout);
void unlock();
直接上代码:
DistributedLock类中拥有的成员变量:
private static CuratorFramework client = null;
private static Logger logger = Logger.getLogger(DistributedLock.class);
protected static CountDownLatch latch = new CountDownLatch(1);
client:为了之后操作zk设置
logger:打印测试日志
latch:lock中要循环等待获取锁,属于一方释放锁和其他再次尝试获取锁之间的沟通桥梁
初始化方法:
public static synchronized void init(String connectString) {
if (client != null)
return; RetryPolicy retryPolicy = new ExponentialBackoffRetry(1000, 3);
client = CuratorFrameworkFactory.builder().connectString(connectString)
.sessionTimeoutMs(10000).retryPolicy(retryPolicy)
.namespace("LockService").build();
client.start(); // 创建锁目录
try {
if (client.checkExists().forPath("/ExclusiveLockDemo") == null) {
client.create().creatingParentsIfNeeded()
.withMode(CreateMode.PERSISTENT)
.withACL(Ids.OPEN_ACL_UNSAFE)
.forPath("/ExclusiveLockDemo");
}
// 创建锁监听
addChildWatcher("/ExclusiveLockDemo");
} catch (Exception e) {
logger.error("ZK服务器连接不上");
throw new RuntimeException("ZK服务器连接不上");
}
}
init方法里做了以下几个事:
1.初始化client
2.创建根目录
3.加根目录的子节点监听(为了unlock)
监听处理函数:
private static void addChildWatcher(String path) throws Exception {
final PathChildrenCache cache = new PathChildrenCache(client, path,
true);
cache.start(StartMode.POST_INITIALIZED_EVENT);
cache.getListenable().addListener(new PathChildrenCacheListener() {
public void childEvent(CuratorFramework client,
PathChildrenCacheEvent event) throws Exception {
if (event.getType().equals(
PathChildrenCacheEvent.Type.INITIALIZED)) { } else if (event.getType().equals(
PathChildrenCacheEvent.Type.CHILD_ADDED)) { } else if (event.getType().equals(
PathChildrenCacheEvent.Type.CHILD_REMOVED)) {
String path = event.getData().getPath();
System.out.println("收到监听"+path);
if(path.contains("ExclusiveLockDemo")){
logger.info("排他锁,收到锁释放通知");
latch.countDown();
}
} else if (event.getType().equals(
PathChildrenCacheEvent.Type.CHILD_UPDATED)) { } }
});
}
主要监听的是子节点的删除行为,当子节点lock删除时,就代表着有节点释放了锁,同时,应该通知等待获取锁的client发起新一轮的获取锁行为,这里用latch控制,latch.countDown()就是通知他们进行下一轮获取,具体代码可以看lock函数的实现。
lock函数:
public void lock() {
while (true) {
try {
client.create().creatingParentsIfNeeded()
.withMode(CreateMode.EPHEMERAL)
.withACL(Ids.OPEN_ACL_UNSAFE)
.forPath("/ExclusiveLockDemo/lock");
logger.info("成功获取到锁");
return;// 如果节点创建成功,即说明获取锁成功
} catch (Exception e) {
logger.info("此次获取锁没有成功");
try {
//如果没有获取到锁,需要重新设置同步资源值
if(latch.getCount()<=0){
latch = new CountDownLatch(1);
}
latch.await();
} catch (InterruptedException e1) {
e1.printStackTrace();
logger.error("", e1);
}
}
} }
lock函数:尝试在 “/ExclusiveLockDemo/lock”创建znode,当有多个客户端同时调用lock()时,有且只有一个client可以成功创建,打印“成功获取到锁”,其他client将会进入catch语句,打印“此次获取没有成功”,然后重置latch,进入阻塞状态(latch.await())。唤醒条件是持有锁的client释放锁。因为是阻塞的获取锁,所以整个函数处于while(true)死循环里。直到某个时间点成功创建锁,方可return。
islock()函数:
public boolean isLocked() {
try {
Stat stat = client.checkExists().forPath("/ExclusiveLockDemo/lock");
return stat==null?false:true;
} catch (Exception e) {
e.printStackTrace();
} return false;
}
这个比较简单,就是判断一下制定path是否存在znode,也就是是否已经有client成功获取到锁并且还没有释放。
unlock函数:
public void unlock() {
try {
if (client.checkExists().forPath("/ExclusiveLockDemo/lock") != null) {
client.delete().forPath("/ExclusiveLockDemo/lock");
}
} catch (Exception e) {
e.printStackTrace();
} }
这个也比较简单,只是删除锁对应的znode节点即可,但是有个很严重的问题,这种删除太草率,你可以释放一个原本不是你获取的锁,这是不符合道理的,只有你获取的锁,你才有资格释放它,所以正规的写法应该是获取锁的时候加权限,然后释放的时候先检验权限,不是你的不能删。
public boolean tryLock() {
try {
client.create().creatingParentsIfNeeded()
.withMode(CreateMode.EPHEMERAL)
.withACL(Ids.OPEN_ACL_UNSAFE)
.forPath("/ExclusiveLockDemo/lock");
logger.info("成功获取到锁");
} catch (Exception e) {
logger.info("获取锁失败");
return false;
} return true;
}
仿照可重入锁的trylock,尝试获取锁,非阻塞,能获取就返回true,被占用就返回false
public boolean tryLock(long timeout) {
if(timeout<= 0L) return false;
final long deadline = System.currentTimeMillis() + timeout;
for(;;){
if(tryLock())
return true;
else{
timeout = deadline - System.currentTimeMillis();
if (timeout <= 0L)
return false;
}
}
}
有获取时间限制的trylock,实现起来还是比较简单的,大家一看都能看懂
分布式锁 ----zookeeper实践 (排它锁)的更多相关文章
- ZooKeeper分布式锁简单实践
ZooKeeper分布式锁简单实践 在分布式解决方案中,Zookeeper是一个分布式协调工具.当多个JVM客户端,同时在ZooKeeper上创建相同的一个临时节点,因为临时节点路径是保证唯一,只要谁 ...
- [Java复习] 分布式锁 Zookeeper Redis
一般实现分布式锁都有哪些方式? 使用 Redis 如何设计分布式锁?使用 Zookeeper 来设计分布式锁可以吗? 这两种分布式锁的实现方式哪种效率比较高? 1. Zookeeper 都有哪些使用场 ...
- zookeeper 实现分布式锁zookeeper 使用 Curator 示例监听、分布式锁
下载地址: http://download.csdn.net/download/ttyyadd/10239642
- Redis、Zookeeper实现分布式锁——原理与实践
Redis与分布式锁的问题已经是老生常谈了,本文尝试总结一些Redis.Zookeeper实现分布式锁的常用方案,并提供一些比较好的实践思路(基于Java).不足之处,欢迎探讨. Redis分布式锁 ...
- 面试必问:分布式锁实现之zk(Zookeeper)
点赞再看,养成习惯,微信搜索[三太子敖丙]关注这个互联网苟且偷生的工具人. 本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试完整考点.资料以及我的 ...
- Zookeeper 分布式锁 (图解+秒懂+史上最全)
文章很长,而且持续更新,建议收藏起来,慢慢读! 高并发 发烧友社群:疯狂创客圈(总入口) 奉上以下珍贵的学习资源: 疯狂创客圈 经典图书 : 极致经典 + 社群大片好评 < Java 高并发 三 ...
- 分布式锁(Zookeeper实现)
分布式锁 分布式锁,这个主要得益于 ZooKeeper 为我们保证了数据的强一致性.锁服务可以分为两类,一个是 保持独占,另一个是 控制时序. 1. 所谓保持独占,就是所有试图来获取这个锁的客户端,最 ...
- Docker 下的Zookeeper以及.ne core 的分布式锁
单节点 1.拉取镜像:docker pull zookeeper 2.运行容器 a.我的容器同一放在/root/docker下面,然后创建相应的目录和文件, mkdir zookeeper cd zo ...
- 整理分布式锁:业务场景&分布式锁家族&实现原理
1.引入业务场景 业务场景一出现: 因为小T刚接手项目,正在吭哧吭哧对熟悉着代码.部署架构.在看代码过程中发现,下单这块代码可能会出现问题,这可是分布式部署的,如果多个用户同时购买同一个商品,就可能导 ...
随机推荐
- SVN_03绿色版
1.首先备份当前安装visualSVN文件的bin目录,万一出错还能反个水.一般默认安装路径是C:\Program Files(x86)VisualSVN\bin 2.然后运行ildasm,Windo ...
- Python实现行列式计算
数学公式: 代码: # 逆序数 def getInversion(numlist): count = 0 for i in range(1,len(numlist)): subscript = num ...
- memoryCache的使用
1 借鉴这篇文章 https://www.cnblogs.com/zuowj/p/8440902.html using System; using System.Collections.Generic ...
- 利用 pandas库读取excel表格数据
利用 pandas库读取excel表格数据 初入IT行业,愿与大家一起学习,共同进步,有问题请指出!! 还在为数据读取而头疼呢,请看下方简洁介绍: 数据来源为国家统计局网站下载: 具体方法 代码: i ...
- openwrt 切换overlay文件系统为根文件系统
http://blog.chinaunix.net/uid-27057175-id-4584360 openwrt的overlayfs 通过/etc/preinit调用 /sbin/mount_roo ...
- 《Python编程:从入门到实践》第四章 操作列表 习题答案
#4.1 pizzas = ['KFC','MDL','DKS'] ''' for pizza in pizzas: print(pizza); ''' for pizza in pizzas: pr ...
- Plot the figure of K-SVCR
clear %% generate data prettySpiral = 0; if ~prettySpiral % generate some random gaussian like data ...
- 理解serverless无服务
理解serverless无服务 阅读目录 一:什么是serverless无服务? 二:与传统模式架构区别? 三:serverless优缺点? 四:使用serverless的应用场景有哪些? 回到顶部 ...
- Eclipse修改项目jdk版本
修改项目jdk版本(@Override的解决方法) (1)Eclipse的Window-Preferences->Java->Compiler(2)Eclipse的Project-> ...
- 缺包与maven
一. 缺包 1. 打开pom.xml 将依赖加入. 2. mvn clean install -DskipTests 3. scp target/lib/****.jar(刚刚的依赖的架包) 服务器地 ...