原生 Zookeeper 实现分布式锁案例

//1.建立分布式锁的结构

java 复制代码
package zkcase2;

import org.apache.zookeeper.*;
import org.apache.zookeeper.data.Stat;

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

public class DistributedLock {

    private final ZooKeeper zooKeeper;
    private final String connection="hadoop102:2181,hadoop103:2181,hadoop104:2181"; //对应你服务器里面安装zk的节点,注意不可以有空格出现
    private final int sessiontimeout=2000;

    private CountDownLatch connect=new CountDownLatch(1);

    private CountDownLatch waitlath=new CountDownLatch(1);

    private String waitpath;
    private String currentnode;

    public DistributedLock() throws IOException, InterruptedException, KeeperException {

        //获取zk
        zooKeeper = new ZooKeeper(connection, sessiontimeout, new Watcher() {
            @Override
            public void process(WatchedEvent watchedEvent) {

                //connect表示如果连接到zk 可以释放
                if (watchedEvent.getState()==Event.KeeperState.SyncConnected){
                    connect.countDown();
                }
                //waitlath需要释放
                if (watchedEvent.getType()==Event.EventType.NodeDeleted && watchedEvent.getPath().equals(waitpath)){
                    waitlath.countDown();
                }
            }
        });
        //等待zk正常连接之后往下进行
        connect.await();
        //判断根节点/locks是否存在
        Stat exists = zooKeeper.exists("/locks", false);
            if (exists==null){
                //如果状态为空说明根节点不存在,那就创建根节点
                zooKeeper.create("/locks","locks".getBytes(), ZooDefs.Ids.OPEN_ACL_UNSAFE,CreateMode.PERSISTENT);


            }

    }

    //创建zk锁

    public void   zklock() throws InterruptedException {
        //创建对应的临时带序号节点

        try {
            currentnode = zooKeeper.create("/locks/" + "seq-", null, ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.EPHEMERAL_SEQUENTIAL);
            //判断当前节点是不是最小的序号节点,如果是获取到锁,如果不是监听序号前一个节点
            List<String> children = zooKeeper.getChildren("/locks", false);

            //如果childern只有一个值那么就获取锁,如果有多个节点,判断谁最小
            if (children.size()==1){
                return;
            }else {
                Collections.sort(children);
                //获取节点名称seq-000000000000
                String thisnode = currentnode.substring("/locks/".length());
                //通过seq-000000000获取当前节点在childen当中的位置
                int indexOf = children.indexOf(thisnode);
                //判断
                if (indexOf==-1){
                    System.out.println("数据异常");
                } else if (indexOf==0) {
                    //只有一个节点直接获取锁
                    return;

                }else {
                    //不是只有一个节点所以需要监听他前一个节点
                    waitpath="/locks/"+children.get(indexOf-1);
                    zooKeeper.getData(waitpath,true,null);
                    //等待监听
                    waitlath.await();
                }
            }

        } catch (KeeperException e) {
            throw new RuntimeException(e);
        }

    }
    //解锁
    public void unzklock(){
        //删除节点
        try {
            zooKeeper.delete(currentnode,-1);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        } catch (KeeperException e) {
            throw new RuntimeException(e);
        }

    }
}

//2.建立测试类

java 复制代码
package zkcase2;

import org.apache.zookeeper.KeeperException;

import java.io.IOException;

public class DistributedLockTest {
    public static void main(String[] args) throws IOException, InterruptedException, KeeperException {
      final   DistributedLock distributedLock1 = new DistributedLock();


        final   DistributedLock distributedLock2 = new DistributedLock();


        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    distributedLock1.zklock();

                    System.out.println("线程1启动,获取到锁");

                    Thread.sleep(5*1000);

                    distributedLock1.unzklock();
                    System.out.println("线程1释放锁");
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        }).start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    distributedLock2.zklock();

                    System.out.println("线程2启动,获取到锁");

                    Thread.sleep(5*1000);

                    distributedLock2.unzklock();
                    System.out.println("线程2释放锁");
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        }).start();
    }
}
相关推荐
春生野草15 分钟前
反射、Tomcat执行
java·开发语言
_日拱一卒1 小时前
LeetCode:207课程表
java·数据结构·算法·leetcode·职场和发展
飞翔中文网1 小时前
Java学习笔记之抽象类与接口(设计思想)
java·笔记·学习
qcx231 小时前
【系统学AI】09 Multi-Agent架构(2026版):从学术理论到工业级实践
java·人工智能·架构·multi-agent·claude agent
Dongwoo Jeong2 小时前
微服务架构(MSA)是如何诞生的?
微服务·云原生·架构
半旧夜夏2 小时前
【保姆级】微服务组件环境搭建(Docker Compose版)
java·linux·spring cloud·微服务·云原生·容器
阿里云云原生2 小时前
实战解析:如何用自然语言驱动混沌工程?Blade AI Agent 实现故障演练全链路自动化
云原生
云烟成雨TD3 小时前
Spring AI 1.x 系列【33】RAG Advisor 组件与四大分层架构
java·人工智能·spring
张忠琳3 小时前
【kubernetes v1.21】(kubelet 1)Kubelet 核心架构与启动流程
云原生·架构·kubernetes·kubelet
不爱编程的小陈3 小时前
事务的进化:从MySQL单机事务到TiDB分布式事务的探究
分布式·mysql·tidb