zookeeper应用之分布式锁

  前一段时间有讨论过用redis来实现分布式锁,讲到setNx不是原子性、redis新的set方法及其误删和守护线程,还为了原子性不得不使用redis的脚本。虽然最终分布式锁的这个效果是实现了,但是,不够优雅。这里讨论一下zookeeper对分布式锁的实现。

  首先说一下zk中节点的类型,共分为四个类型:持久节点、临时节点和持久有序节点、临时有序节点。
  什么是持久节点呢?字面意思,就是持久化的节点,当创建持久节点后,客户端断开了和服务器的链接,持久节点仍旧存在,与之相反的就是临时节点了,临时节点会在客户端断开链接后消失(客户端主动自刎或者zk服务器清理门户)。
  什么是顺序节点呢?先说下非顺序节点,比如节点“lock”,你只可以创建一次,创建第二次就会提示你“NodeExistsException”

Exception in thread "main" org.apache.zookeeper.KeeperException$NodeExistsException: KeeperErrorCode = NodeExists for /lock

  假如使用的是顺序节点呢,上面这个异常就不会出现了,顺序节点会在节点名字后面追加一个序号,如果你使用“/lock”作为顺序节点,实际节点路径是“/lock0000000001”,十位数量的坑,绝对够用了。

  再来说一下zk的数据结构.zk是树形的数据结构,
zookeeper应用之分布式锁_第1张图片
我们可以利用最左节点这个特性来实现分布式锁:如果有多个请求过来,会依次挂在指定节点下,我们每次取最左边的那个节点,执行完以后删除节点,让第二左节点接棒成为最左节点。

  下面就是具体的代码了:
先看下Main.java,这个是入口

public class Main {
    private static final Logger LOG = LoggerFactory.getLogger(Main.class);
    private static final int SESSION_TIMEOUT = 10000;
    private static final String CONNECTION_STRING = "192.168.2.201:2181,192.168.2.201:2182,192.168.2.201:2183";

    static final String GROUP_PATH = "/lockParent";
    static final String SUB_PATH = GROUP_PATH + "/lock";


    public static void main(String[] args) {
        for (int i = 0; i < DistributedLock.THREAD_NUM; i++) {
            final int threadId = i + 1;
            new Thread() {
                @Override
                public void run() {
                    try {
                        DistributedLock dc = new DistributedLock(threadId, GROUP_PATH, SUB_PATH);
                        // 1.实例化zk客户端
                        dc.createConnection(CONNECTION_STRING, SESSION_TIMEOUT);
                        // 2.创建公共的lock父节点(有/无视 无/创建)
                        dc.createParent(GROUP_PATH, "该节点由线程" + threadId + "创建", true);
                        // 3.创建子节点
                        dc.createChilden();
                        // 4.判断当前节点是不是罪左节点,是的话则成功获得锁
                        if (dc.checkMinPath()) {
                            dc.getLockSuccess();
                        }
                    } catch (Exception e) {
                        LOG.error("【第" + threadId + "个线程】 抛出的异常:");
                        e.printStackTrace();
                    }
                }
            }.start();
        }

        try {
            DistributedLock.threadSemaphore.await();
            LOG.info("所有线程运行结束!");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

    }
}

还有具体的执行类 DistributedLock.java

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.zookeeper.*;
import org.apache.zookeeper.data.Stat;

import java.util.List;
import java.io.IOException;
import java.util.Collections;
import java.util.concurrent.CountDownLatch;

public class DistributedLock implements Watcher {
    private static final Logger LOG = LoggerFactory.getLogger(DistributedLock.class);

    /** 父lock节点,持久有序节点,本demo所有的锁都是在这个节点下面 */
    private String groupPath;
    /** 具体的锁节点,隶属于groupPath下 */
    private String subPath;
    /** 当前线程信息,做日志输出时候用到 */
    private String LOG_PREFIX_OF_THREAD;

    /** zk的客户端连接 */
    private ZooKeeper zk = null;
    /** 本节点的path */
    private String selfPath;
    /** pre节点的path */
    private String waitPath;
    /** 模拟的请求个数 */
    static final int THREAD_NUM = 3;
    /** 确保连接zk成功才开始工作,避免抛出连接丢失异常 
     * org.apache.zookeeper.KeeperException$ConnectionLossException: KeeperErrorCode = ConnectionLoss for /lockParent
     */
    private CountDownLatch connectedSemaphore = new CountDownLatch(1);
    /** 确保所有线程运行结束; */
    static final CountDownLatch threadSemaphore = new CountDownLatch(THREAD_NUM);

    public DistributedLock(int threadId,String groupPath,String subPath) {
        this.groupPath = groupPath;
        this.subPath = subPath;
        LOG_PREFIX_OF_THREAD = "【第" + threadId + "个线程】";
    }

    /**
     * 获取锁成功
     */
    public void getLockSuccess() throws KeeperException, InterruptedException {
        if (zk.exists(this.selfPath, false) == null) {
            LOG.error(LOG_PREFIX_OF_THREAD + "本节点已不在了...");
            return;
        }
        LOG.info(LOG_PREFIX_OF_THREAD + "获取锁成功,赶紧干活!");
        Thread.sleep(5000);
        LOG.info(LOG_PREFIX_OF_THREAD + "删除本节点:" + selfPath);
        zk.delete(this.selfPath, -1);
        releaseConnection();
        threadSemaphore.countDown();
    }

    /**
     * 检查自己是不是最小的节点
     * @return
     */
    public boolean checkMinPath() throws KeeperException, InterruptedException {
        List subNodes = zk.getChildren(groupPath, false);
        Collections.sort(subNodes);
        int index = subNodes.indexOf(selfPath.substring(groupPath.length() + 1));
        switch (index) {
            case -1: {
                LOG.error(LOG_PREFIX_OF_THREAD + "父lock节点下找不到本节点:" + selfPath);
                return false;
            }
            case 0: {
                LOG.info(LOG_PREFIX_OF_THREAD + "本节点是最左节点:" + selfPath);
                return true;
            }
            default: {
                this.waitPath = groupPath + "/" + subNodes.get(index - 1);
                LOG.info(LOG_PREFIX_OF_THREAD + "获取子节点中,排在我前面的节点是:" + waitPath);
                try {
                    zk.getData(waitPath, true, new Stat());
                } catch (KeeperException e) {
                    if (zk.exists(waitPath, false) == null) {
                        LOG.info(LOG_PREFIX_OF_THREAD + "子节点中,排在我前面的" + waitPath + "已失踪,幸福来得太突然?");
                        return checkMinPath();
                    } else {
                        throw e;
                    }
                }
                return false;
            }
        }

    }

    /**
     * 创建ZK连接
     * @param connectString    ZK服务器地址列表
     * @param sessionTimeout   Session超时时间
     */
    public void createConnection(String connectString, int sessionTimeout) throws IOException, InterruptedException {
        zk = new ZooKeeper(connectString, sessionTimeout, this);
        connectedSemaphore.await();
    }

    /**
     * 创建子节点
     * @return
     */
    public void createChilden() throws KeeperException, InterruptedException {
        selfPath = zk.create(subPath, null, ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
        LOG.info(LOG_PREFIX_OF_THREAD + "创建锁路径:" + selfPath);
    }

    /**
     * 创建节点
     * @param path 节点path
     * @param data 初始数据内容
     * @return
     */
    public void createParent(String path, String data, boolean needWatch)
            throws KeeperException, InterruptedException {
        if (null == zk.exists(path, needWatch)) {
            String createResult = this.zk.create(path, data.getBytes(), ZooDefs.Ids.OPEN_ACL_UNSAFE,
                    CreateMode.PERSISTENT);
            LOG.info(LOG_PREFIX_OF_THREAD + "节点创建成功, Path: " + createResult + ", content: " + data);
        }
    }

    /**
     * 关闭ZK连接
     */
    public void releaseConnection() {
        if (this.zk != null) {
            try {
                this.zk.close();
            } catch (InterruptedException e) {
            }
        }
        LOG.info(LOG_PREFIX_OF_THREAD + "释放连接");
    }

    public void process(WatchedEvent event) {
        if (event == null) {
            return;
        }
        Event.KeeperState keeperState = event.getState();
        Event.EventType eventType = event.getType();
        if (Event.KeeperState.SyncConnected == keeperState) {
            if (Event.EventType.None == eventType) {
                LOG.info(LOG_PREFIX_OF_THREAD + "成功连接上ZK服务器");
                connectedSemaphore.countDown();
            } else if (event.getType() == Event.EventType.NodeDeleted && event.getPath().equals(waitPath)) {
                LOG.info(LOG_PREFIX_OF_THREAD + "收到情报,排我前面的家伙已挂,我是不是可以出山了?");
                try {
                    if (checkMinPath()) {
                        getLockSuccess();
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        } else if (Event.KeeperState.Disconnected == keeperState) {
            LOG.info(LOG_PREFIX_OF_THREAD + "与ZK服务器断开连接");
        } else if (Event.KeeperState.AuthFailed == keeperState) {
            LOG.info(LOG_PREFIX_OF_THREAD + "权限检查失败");
        } else if (Event.KeeperState.Expired == keeperState) {
            LOG.info(LOG_PREFIX_OF_THREAD + "会话失效");
        }
    }
}   

运行Main类,可以清楚的看到三个线程先后获取锁、等待获取锁的信息

2018-06-23 23:54:56,656 - 【第3个线程】成功连接上ZK服务器
2018-06-23 23:54:56,867 - 【第3个线程】创建锁路径:/lockParent/lock0000000007
2018-06-23 23:54:56,923 - 【第3个线程】本节点是最左节点:/lockParent/lock0000000007
2018-06-23 23:54:56,936 - 【第3个线程】获取锁成功,赶紧干活!
2018-06-23 23:54:59,604 - 【第2个线程】成功连接上ZK服务器
2018-06-23 23:54:59,612 - 【第1个线程】成功连接上ZK服务器
2018-06-23 23:54:59,686 - 【第2个线程】创建锁路径:/lockParent/lock0000000008
2018-06-23 23:54:59,698 - 【第1个线程】创建锁路径:/lockParent/lock0000000009
2018-06-23 23:54:59,706 - 【第1个线程】获取子节点中,排在我前面的节点是:/lockParent/lock0000000008
2018-06-23 23:54:59,709 - 【第2个线程】获取子节点中,排在我前面的节点是:/lockParent/lock0000000007
2018-06-23 23:55:01,937 - 【第3个线程】删除本节点:/lockParent/lock0000000007
2018-06-23 23:55:01,963 - 【第2个线程】收到情报,排我前面的家伙已挂,我是不是可以出山了?
2018-06-23 23:55:01,979 - 【第2个线程】本节点是最左节点:/lockParent/lock0000000008
2018-06-23 23:55:01,993 - 【第2个线程】获取锁成功,赶紧干活!
2018-06-23 23:55:02,029 - 【第3个线程】释放连接
2018-06-23 23:55:06,993 - 【第2个线程】删除本节点:/lockParent/lock0000000008
2018-06-23 23:55:07,008 - 【第1个线程】收到情报,排我前面的家伙已挂,我是不是可以出山了?
2018-06-23 23:55:07,032 - 【第1个线程】本节点是最左节点:/lockParent/lock0000000009
2018-06-23 23:55:07,039 - 【第2个线程】释放连接
2018-06-23 23:55:07,062 - 【第1个线程】获取锁成功,赶紧干活!
2018-06-23 23:55:12,063 - 【第1个线程】删除本节点:/lockParent/lock0000000009
2018-06-23 23:55:12,107 - 【第1个线程】释放连接
2018-06-23 23:55:12,107 - 所有线程运行结束!

假如觉得上面的实现不够简洁,curator开源项目提供zookeeper分布式锁实现
这是maven依赖


    org.apache.curator
    curator-recipes
    4.0.0
   

具体代码:

public static void main(String[] args) throws Exception {
    //创建zookeeper的客户端
    RetryPolicy retryPolicy = new ExponentialBackoffRetry(1000, 3);
    CuratorFramework client = CuratorFrameworkFactory.newClient("10.21.41.181:2181,10.21.42.47:2181,10.21.49.252:2181", retryPolicy);
    client.start();
    //创建分布式锁, 锁空间的根节点路径为/curator/lock
    InterProcessMutex mutex = new InterProcessMutex(client, "/curator/lock");
    mutex.acquire();
    //获得了锁, 进行业务流程
    System.out.println("Enter mutex");
    //完成业务流程, 释放锁
    mutex.release();
    //关闭客户端
    client.close();
}   

短短的十行代码就能实现锁的效果,只需要在acquire和release代码中间进行我们的业务操作即可,十分简洁明了

你可能感兴趣的:(中间件/缓存)