基于Zookeeper的分布式共享锁
实现原理
- 基于Zookeeper、Lock实现的分布共享式锁
- 构造初始化Zookeeper连接
- 在lock中尝试获取锁(tryLock)
- 首先创建当前连接的节点
- 获取所有相关节点,并排序
- 若当前为只有一个节点或为最小值,直接返回获取锁成功
- 否则获取前一个节点,监听事件,让当前节点进入等待状态
- 如果监听到事件为上删除事件,释放锁
- 删除节点,释放资源
代码实现
@Data
@Slf4j
public class MyZkDistributedLock implements Lock, Watcher {
// 超时时间
private static final int SESSION_TIMEOUT = 5000;
// zookeeper server列表
private String hosts;
private String groupNode = "locks";
private String subNode = "sub";
private String lockName;
private ZooKeeper zk;
// 当前client创建的子节点
private String thisPath;
// 当前client等待的子节点
private String waitPath;
private List exceptionList = new ArrayList<>();
private CountDownLatch latch = new CountDownLatch(1);
public MyZkDistributedLock(String hosts, String lockName) {
this.hosts = hosts;
this.lockName = lockName;
try {
// 连接zookeeper
zk = new ZooKeeper(hosts, SESSION_TIMEOUT, this);
Stat stat = zk.exists(groupNode, false);
if (stat == null) {
// 如果根节点不存在,则创建根节点
zk.create(groupNode, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE,
CreateMode.PERSISTENT);
}
} catch (IOException e) {
log.error("Zk连接异常", e);
exceptionList.add(e);
} catch (InterruptedException e) {
log.error("Zk连接异常", e);
exceptionList.add(e);
} catch (KeeperException e) {
log.error("Zk连接异常", e);
exceptionList.add(e);
}
}
@Override
public void lock() {
if (exceptionList.size() > 0) {
throw new LockException(exceptionList.get(0));
}
try {
if (this.tryLock()) {
log.info("------------>线程:{},锁:{},获得", Thread.currentThread().getName(), lockName);
return;
} else {
// 等待锁
this.latch.await();
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
@Override
public void lockInterruptibly() throws InterruptedException {
}
@Override
public boolean tryLock() {
try {
String splitStr = "_lock_";
if (lockName.contains(splitStr)) {
throw new MyZkDistributedLock.LockException("锁名有误");
}
// 创建子节点
thisPath = zk
.create("/" + groupNode + "/" + lockName + subNode + splitStr, null,
Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
// 注意, 没有必要监听"/locks"的子节点的变化情况
List childrenNodes = zk.getChildren("/" + groupNode, false);
// 取出所有lockName的锁
List lockObjects = new ArrayList();
for (String node : childrenNodes) {
String _node = node.split(splitStr)[0];
if (_node.equals(lockName)) {
lockObjects.add(node);
}
}
// 列表中只有一个子节点, 那肯定就是thisPath, 说明client获得锁
if (lockObjects.size() == 1) {
return true;
} else {
String thisNode = thisPath.substring(("/" + groupNode + "/").length());
// 排序
Collections.sort(lockObjects);
int index = lockObjects.indexOf(thisNode);
if (index == -1) {
// never happened
} else if (index == 0) {
// inddx == 0, 说明thisNode在列表中最小, 获得锁
return true;
} else {
// 获得排名比thisPath前1位的节点
this.waitPath = "/" + groupNode + "/" + lockObjects.get(index - 1);
// 在waitPath上注册监听器, 当waitPath被删除时, zookeeper会回调监听器的process方法
zk.getData(waitPath, true, new Stat());
}
}
} catch (Exception e) {
}
return false;
}
@Override
public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
return tryLock();
}
@Override
public void unlock() {
try {
log.info("释放锁 {}", thisPath);
zk.delete(thisPath, -1);
thisPath = null;
zk.close();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (KeeperException e) {
e.printStackTrace();
}
}
@Override
public Condition newCondition() {
return null;
}
@Override
public void process(WatchedEvent event) {
try {
// 发生了waitPath的删除事件
if (event.getType() == EventType.NodeDeleted && event.getPath().equals(waitPath)) {
this.latch.countDown();
}
} catch (Exception e) {
e.printStackTrace();
}
}
public class LockException extends RuntimeException {
private static final long serialVersionUID = 1L;
public LockException(String e) {
super(e);
}
public LockException(Exception e) {
super(e);
}
}
}
总结
总体来说,实现并不难,我认为主要就是排序号监听上一个节点的删除事件,依此类推,最后实现所有节点的监听
代码地址
源码GitHub地址
参考
分布式锁与实现(二)——基于ZooKeeper实现