`

Zookeeper分布式锁实现

 
阅读更多

引入zkclient包

<dependency>
    <groupId>com.101tec</groupId>
    <artifactId>zkclient</artifactId>
    <version>0.10</version>
</dependency>

1.定义分布式锁接口

package com.springboot.zookeeper.distribute;
import java.util.concurrent.TimeUnit;
/**
 * Created by washingtin on 2019/6/24.
 */
public interface DistributedLock {
    /**
     * 获取锁,如果没有得到锁就一直等待
     *
     * @throws Exception
     */
public void acquire() throws Exception;
/**
     * 获取锁,如果没有得到锁就一直等待直到超时
     *
     * @param time 超时时间
     * @param unit time参数时间单位
     *
     * @return 是否获取到锁
     * @throws Exception
     */
public boolean acquire(long time, TimeUnit unit) throws Exception;
/**
     * 释放锁
     *
     * @throws Exception
     */
public void release() throws Exception;
}

2.分布式锁的实现细节

  获取分布式锁的重点逻辑在于BaseDistributedLock,实现了基于Zookeeper实现分布式锁的细节。

  

package com.springboot.zookeeper.distribute;
import org.I0Itec.zkclient.IZkDataListener;
import org.I0Itec.zkclient.ZkClient;
import org.I0Itec.zkclient.exception.ZkNoNodeException;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
/**
 * 实现zookeeper的分布式锁
 * Created by washingtin on 2019/6/24.
 */
public class BaseDistributedLock {

    private final ZkClient client;
//用于保存Zookeeper中实现分布式锁的节点,例如/locker节点,该节点是个持久节点,在该节点下面创建临时顺序节点来实现分布式锁
private final String basePath ;
//同basePath变量一样
private final String path ;
//锁名称前缀,/locker下创建的顺序节点,例如以lock-开头,这样便于过滤无关节点
private final String lockName;
//最大重试次数
private static final Integer MAX_RETRY_COUNT = 10;
    public BaseDistributedLock(ZkClient client, String path, String lockName) {
        this.client = client;
        this.basePath = path;
        this.path = path.concat("/").concat(lockName);
        this.lockName = lockName;
}

    /**
     * 删除节点
     *
     * @param path
* @throws Exception
     */
private void deletePath(String path) throws Exception {
        client.delete(path);
}

    /**
     * 创建临时顺序节点
     *
     * @param client Zookeeper客户端
     * @param path 节点路径
     * @return
* @throws Exception
     */
private String createEphemeralSequential(ZkClient client, String path) throws Exception {
        return client.createEphemeralSequential(path, null);
}

    /**
     * 获取锁的核心方法
     *
     * @param startMillis 当前系统时间
     * @param millisToWait 超时时间
     * @param path
* @return
* @throws Exception
     */
private boolean waitToLock(long startMillis, Long millisToWait, String path) throws Exception {
        //获取锁标志
boolean haveTheLock = false;
//删除锁标志
boolean doDelete = false;
        try {
            while (!haveTheLock) {
                // 获取/locker节点下的所有顺序节点,并且从小到大排序
List<String> children = getSortedChildren();
// 获取子节点,如:/locker/node_0000000003返回node_0000000003
String sequenceNodeName = path.substring(basePath.length() + 1);
// 计算刚才客户端创建的顺序节点在locker的所有子节点中排序位置,如果是排序为0,则表示获取到了锁
int ourIndex = children.indexOf(sequenceNodeName);
/*
                 * 如果在getSortedChildren中没有找到之前创建的[临时]顺序节点,这表示可能由于网络闪断而导致
                 * Zookeeper认为连接断开而删除了我们创建的节点,此时需要抛出异常,让上一级去处理
                 * 上一级的做法是捕获该异常,并且执行重试指定的次数,见后面的 attemptLock方法
                 */
if (ourIndex < 0) {
                    throw new ZkNoNodeException("节点没有找到: " + sequenceNodeName);
}

                // 如果当前客户端创建的节点在locker子节点列表中位置大于0,表示其它客户端已经获取了锁
                // 此时当前客户端需要等待其它客户端释放锁
boolean isGetTheLock = ourIndex == 0; //是否得到锁
                // 如何判断其它客户端是否已经释放了锁?从子节点列表中获取到比自己次小的那个节点,并对其建立监听
                // 获取比自己次小的那个节点,如:node_0000000002
String pathToWatch = isGetTheLock ? null : children.get(ourIndex - 1);
                if (isGetTheLock) {
                    haveTheLock = true;
} else {
                    // 如果次小的节点被删除了,则表示当前客户端的节点应该是最小的了,所以使用CountDownLatch来实现等待
String previousSequencePath = basePath.concat("/").concat(pathToWatch);
                    final CountDownLatch latch = new CountDownLatch(1);
                    final IZkDataListener previousListener = new IZkDataListener() {
                        /**
                         * 监听指定节点删除时触发该方法
                         */
public void handleDataDeleted(String dataPath)
                                throws Exception {
                            // 次小节点删除事件发生时,让countDownLatch结束等待
                            // 此时还需要重新让程序回到while,重新判断一次!
latch.countDown();
}

                        /**
                         * 监听指定节点的数据发生变化触发该方法
                         *
                         */
public void handleDataChange(String dataPath,
Object data) throws Exception {

                        }

                    };
                    try {
                        // 如果节点不存在会出现异常
                        // 监听比自己次小的那个节点
client.subscribeDataChanges(previousSequencePath, previousListener);
//发生超时需要删除节点
if (millisToWait != null) {
                            millisToWait -= (System.currentTimeMillis() - startMillis);
startMillis = System.currentTimeMillis();
                            if (millisToWait <= 0) {
                                doDelete = true; // timed out - delete our node
break;
}

                            latch.await(millisToWait, TimeUnit.MICROSECONDS);
} else {
                            latch.await();
}

                    } catch (ZkNoNodeException e) {
                        // ignore
} finally {
                        client.unsubscribeDataChanges(previousSequencePath, previousListener);
}
                }
            }
        } catch (Exception e) {
            // 发生异常需要删除节点
doDelete = true;
            throw e;
} finally {
            // 如果需要删除节点
if (doDelete) {
                deletePath(path);
}
        }
        return haveTheLock;
}

    private String getLockNodeNumber(String str, String lockName) {
        int index = str.lastIndexOf(lockName);
        if (index >= 0) {
            index += lockName.length();
            return index <= str.length() ? str.substring(index) : "";
}
        return str;
}

    /**
     * 获取parentPath节点下的所有顺序节点,并且从小到大排序
     *
     * @return
* @throws Exception
     */
private List<String> getSortedChildren() throws Exception {
        try {
            List<String> children = client.getChildren(basePath);
Collections.sort(children, new Comparator<String>() {
                public int compare(String lhs, String rhs) {
                    return getLockNodeNumber(lhs, lockName).compareTo(
                            getLockNodeNumber(rhs, lockName));
}
            });
            return children;
} catch (ZkNoNodeException e) {
            client.createPersistent(basePath, true); //创建锁持久节点
return getSortedChildren();
}
    }

    /**
     * 释放锁
     *
     * @param lockPath
* @throws Exception
     */
protected void releaseLock(String lockPath) throws Exception {
        deletePath(lockPath);
}

    /**
     * 尝试获取锁
     *
     * @param time
* @param unit
* @return
* @throws Exception
     */
protected String attemptLock(long time, TimeUnit unit) throws Exception {
        final long startMillis = System.currentTimeMillis();
        final Long millisToWait = (unit != null) ? unit.toMillis(time) : null;
String ourPath = null;
        boolean hasTheLock = false; //获取锁标志
boolean isDone = false; //是否完成得到锁
int retryCount = 0; //重试次数
        // 网络闪断需要重试一试
while (!isDone) {
            isDone = true;
            try {
                // createLockNode用于在locker(basePath持久节点)下创建客户端要获取锁的[临时]顺序节点
ourPath = createEphemeralSequential(client, path);
/**
                 * 该方法用于判断自己是否获取到了锁,即自己创建的顺序节点在locker的所有子节点中是否最小
                 * 如果没有获取到锁,则等待其它客户端锁的释放,并且稍后重试直到获取到锁或者超时
                 */
hasTheLock = waitToLock(startMillis, millisToWait, ourPath);
} catch (ZkNoNodeException e) {
                if (retryCount++ < MAX_RETRY_COUNT) {
                    isDone = false;
} else {
                    throw e;
}
            }
        }

        System.out.println(ourPath + "锁获取" + (hasTheLock ? "成功" : "失败"));
        if (hasTheLock) {
            return ourPath;
}

        return null;
}
}

3.定义一个简单的互斥锁

  

package com.springboot.zookeeper.distribute;
import org.I0Itec.zkclient.ZkClient;
import java.io.IOException;
import java.util.concurrent.TimeUnit;
/**
 * Created by washingtin on 2019/6/24.
 */
public class SimpleDistributedLock extends BaseDistributedLock implements DistributedLock {

    /*
     * 用于保存Zookeeper中实现分布式锁的节点,如名称为locker:/locker,
     * 该节点应该是持久节点,在该节点下面创建临时顺序节点来实现分布式锁
     */
private final String basePath;
/*
     * 锁名称前缀,locker下创建的顺序节点例如都以lock-开头,这样便于过滤无关节点
     * 这样创建后的节点类似:lock-00000001,lock-000000002
     */
private static final String LOCK_NAME = "lock-";
/*
     * 用于保存某个客户端在locker下面创建成功的顺序节点,用于后续相关操作使用(如判断)
     *
     */
private String ourLockPath;
/**
     * 传入Zookeeper客户端连接对象,和basePath
     *
     * @param client
*            Zookeeper客户端连接对象
     * @param basePath
*            basePath是一个持久节点
     */
public SimpleDistributedLock(ZkClient client, String basePath) {
        /*
         * 调用父类的构造方法在Zookeeper中创建basePath节点,并且为basePath节点子节点设置前缀
         * 同时保存basePath的引用给当前类属性
         */
super(client, basePath, LOCK_NAME);
        this.basePath = basePath;
}

    /**
     * 用于获取锁资源,通过父类的获取锁方法来获取锁
     *
     * @param time 获取锁的超时时间
     * @param unit 超时时间单位
     *
     * @return 是否获取到锁
     * @throws Exception
     */
private boolean internalLock(long time, TimeUnit unit) throws Exception {
        // 如果ourLockPath不为空则认为获取到了锁,具体实现细节见attemptLock的实现
ourLockPath = attemptLock(time, unit);
        return ourLockPath != null;
}

    /**
     * 获取锁,如果没有得到锁就一直等待
     *
     * @throws Exception
     */
public void acquire() throws Exception {
        // -1表示不设置超时时间,超时由Zookeeper决定
if (!internalLock(-1, null)) {
            throw new IOException("连接丢失!在路径:'" + basePath + "'下不能获取锁!");
}
    }

    /**
     * 获取锁,如果没有得到锁就一直等待直到超时
     *
     * @param time 超时时间
     * @param unit time参数时间单位
     *
     * @return 是否获取到锁
     * @throws Exception
     */
public boolean acquire(long time, TimeUnit unit) throws Exception {
        return internalLock(time, unit);
}


    /**
     * 释放锁
     */
public void release() throws Exception {
        releaseLock(ourLockPath);
System.out.println(ourLockPath + "锁已释放...");
}
}

4.Test

  

package com.springboot.zookeeper;import com.springboot.zookeeper.distribute.SimpleDistributedLock;
import org.I0Itec.zkclient.ZkClient;
import org.junit.Test;
/**
 * Created by washingtin on 2019/6/24.
 */
public class DistributeLockTest {

    @Test
public void distributeLockTest() throws Exception {
        ZkClient zkClient = new ZkClient("127.0.0.1:2181", 3000);
SimpleDistributedLock simple = new SimpleDistributedLock(zkClient, "/locker");
        for (int i = 0; i < 10; i++) {
            try {
                simple.acquire();
System.out.println("正在进行运算操作:" + System.currentTimeMillis());
} catch (Exception e) {
                e.printStackTrace();
} finally {
                simple.release();
System.out.println("=================\r\n");
}
        }
    }
}

 

 

分享到:
评论

相关推荐

    zookeeper分布式锁实现和客户端简单实现

    **Zookeeper的分布式锁实现原理** 1. **节点创建与监视**: Zookeeper允许客户端创建临时节点,这些节点会在客户端断开连接时自动删除。分布式锁的实现通常会为每个请求创建一个临时顺序节点,按照创建的顺序形成一...

    基于zookeeper的分布式锁实现demo

    **Zookeeper分布式锁的关键特性包括:** 1. **顺序一致性:** Zookeeper中的节点被创建顺序是全局唯一的,这有助于实现锁的唯一性。 2. **原子性:** 创建和删除节点的操作在Zookeeper中都是原子性的,这保证了...

    基于zookeeper的分布式锁简单实现

    这时,Zookeeper,一个高可用的分布式协调服务,常被用来实现分布式锁。 Zookeeper由Apache基金会开发,它提供了一种可靠的分布式一致性服务,包括命名服务、配置管理、集群同步、领导者选举等功能。Zookeeper基于...

    使用ZooKeeper实现分布式锁

    在处理订单生成的场景中,我们可以这样应用ZooKeeper分布式锁: 1. 当用户发起订单请求时,服务端会尝试在ZooKeeper上创建一个临时顺序节点。 2. 如果创建成功,服务端会检查当前最小序号的节点是否是自己创建的。...

    一文彻底理解ZooKeeper分布式锁的实现原理

    《彻底理解ZooKeeper分布式锁实现原理》 ZooKeeper,简称zk,作为一个高可用的分布式协调服务,常被用于构建分布式系统中的各种组件,如分布式锁。在本篇文章中,我们将深入探讨如何利用Curator这个流行的开源框架...

    zookeeper做分布式锁

    分布式锁是解决多节点系统中同步问题的一种常见技术,ZooKeeper,由Apache基金会开发的分布式协调服务,常被用于实现高效可靠的分布式锁。本文将深入探讨如何利用ZooKeeper来构建分布式锁,并讨论其背后的关键概念和...

    zookeeper 分布式锁的实现1

    Zookeeper 的分布式锁实现依赖于其核心特性: 1. **节点的互斥性**:Zookeeper 允许客户端创建临时节点(EPHEMERAL)。这些节点在客户端会话结束(比如客户端宕机)时会被自动删除。通过竞争创建特定路径下的临时...

    C#基于zookeeper分布式锁的实现源码

    总之,C#中基于ZooKeeper的分布式锁实现涉及对ZooKeeper的操作,包括创建临时顺序节点、监听节点变化以及正确释放锁。这样的实现方式保证了在分布式环境下的并发控制和数据一致性,同时具备良好的扩展性和容错性。...

    基于zookeeper实现分布式锁

    zooKeeper是一个分布式的,开放源码的分布式应用程序协调服务,是集群的管理者。提供了文件系统和通知机制。...在开发项目的过程中,很多大型项目都是分布式部署的,那么我们现在使用zookeeper实现一个分布式锁。

    zookeeper分布式锁实例源码

    在这个场景下,我们将关注ZooKeeper如何实现分布式锁,特别是不可重入锁、可重入锁以及可重入读写锁的概念与实践。 首先,我们要理解什么是分布式锁。在多节点并发访问共享资源时,分布式锁能确保同一时刻只有一个...

    从Paxos到Zookeeper分布式一致性原理与实践PDF

    《从Paxos到Zookeeper分布式一致性原理与实践》是一本深入探讨分布式系统一致性问题的著作,其中重点讲解了Paxos算法与Zookeeper在实际应用中的理论与实践。Paxos是分布式计算领域中著名的共识算法,为解决分布式...

    zookeeper分布式锁

    Zookeeper分布式锁的工作原理: 1. **会话和临时节点**:Zookeeper支持两种类型的节点,持久节点和临时节点。临时节点在客户端会话失效(例如,客户端崩溃或网络断开)时会被自动删除,这为实现分布式锁提供了一个...

    从PAXOS到ZOOKEEPER分布式一致性原理与实践

    2. 分布式锁:利用ZNODE的创建、删除操作,实现跨节点的互斥访问控制。 3. 配置管理:集中式存储和分发系统的配置信息,保证所有节点访问的配置是一致的。 4. 命名服务:为分布式组件提供全局唯一的ID或名称,简化...

    从Paxos到Zookeeper分布式一致性原理与实践 + ZooKeeper-分布式过程协同技术详解 pdf

    《从Paxos到Zookeeper分布式一致性原理与实践》与《ZooKeeper-分布式过程协同技术详解》这两本书深入探讨了分布式系统中的一个重要概念——一致性,以及如何通过ZooKeeper这一工具来实现高效的分布式协同。...

    zookeeper实现分布式锁

    在程序开发过程中不得不考虑的就是并发问题。在java中对于同一个jvm而言,jdk已经提供了lock和同步等。但是在分布式情况下,往往存在多个进程对一些资源产生竞争...分布式锁顾明思议就是可以满足分布式情况下的并发锁。

    从Paxos到Zookeeper分布式一致性原理与实践包括源码

    Zookeeper基于Paxos和其他一致性算法的实现,为分布式应用程序提供了命名服务、配置管理、分布式锁、群组服务等功能。Zookeeper通过ZNode(类似于文件系统的节点)来存储和操作数据,并采用观察者模式来实时监控数据...

    ZooKeeper分布式过程协同技术详解_new.pdf

    此外,书中还会深入探讨ZooKeeper在实际应用场景中的最佳实践,如如何利用ZooKeeper进行服务发现、实现分布式锁、构建分布式队列等。通过实例分析,读者可以更好地掌握ZooKeeper在分布式系统中的作用和价值。 在...

    zk分布式锁1

    ZooKeeper是一个广泛使用的分布式锁实现方案,本文将对ZooKeeper分布式锁进行详细的介绍。 什么是分布式锁 分布式锁是指在分布式系统中,多个节点之间对共享资源的访问控制机制。分布式锁可以确保在分布式环境中,...

    springboot zookeeper 分布式锁

    以下是一个简单的基于Zookeeper的分布式锁实现示例: ```java @Service public class ZookeeperDistributedLock { @Autowired private CuratorFramework curatorFramework; private String lockPath = "/...

    浅谈Java(SpringBoot)基于zookeeper的分布式锁实现

    Java(SpringBoot)基于zookeeper的分布式锁实现 本文主要介绍了Java(SpringBoot)基于zookeeper的分布式锁实现,通过示例代码详细介绍了分布式锁的实现过程,对大家的学习或者工作具有一定的参考学习价值。 分布式锁...

Global site tag (gtag.js) - Google Analytics