原生 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();
    }
}
相关推荐
SelectDB3 小时前
秒级弹性、最高降本 70%:SelectDB Serverless 如何重塑云数仓资源效率
大数据·后端·云原生
程序猿大帅4 小时前
别再只当调包侠了:用 Spring AI 落地 Function Calling,我被大模型硬生生砸出了三个大坑
java
程序员晓琪5 小时前
约定大于配置:基于 Java 包名自动生成 API 版本路由的最佳实践
java·spring boot·后端
Flittly5 小时前
【AgentScope Java新手村系列】(11)中断与恢复
java·spring boot·spring
众少成多积小致巨5 小时前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
东坡白菜5 小时前
破局全栈:前端开发的Java入门实战记录—JPA(2)
java·后端
SimonKing12 小时前
艹,维护AI写的代码,我心态崩了......
java·后端·程序员
用户2986985301412 小时前
Java Word 文档样式进阶:段落与文本背景色设置完全指南
java·后端
小bo波1 天前
从"任意文件复制"深挖Java I/O:字符流与字节流的本质抉择
java·nio·io流·后端开发·文件复制
nanxun8862 天前
记一次诡异的 Docker 容器"串包"故障排查
java