基于文件系统分布式锁原理

分布式锁:在一个公共的存储服务上打上一个标记,如Redis的setnx命令,是''先到先得''方式获得锁,ZooKeeper有点像下面的demo,比较大小的方式判决谁获得锁。

java 复制代码
package com.ldj.mybatisflex.demo;

import java.util.*;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;

/**
 * User: ldj
 * Date: 2025/1/12
 * Time: 1:50
 * Description: 基于文件系统制作的分布式锁demo
 */
public class LockDemo {

    public static void main(String[] args) throws InterruptedException {

        //模拟文件存储系统
        List<String> files = new CopyOnWriteArrayList<>();
        List<Integer> hashCodes = new ArrayList<>();

        //使用3个线程模拟3台服务往文件存储系统写消息
        writeMsg(files, hashCodes);
        System.out.println("result:" + hashCodes.toString());

        //构建hashCode与IP的对映关系
        Map<String, String> map = new LinkedHashMap<>();
        for (int i = 0; i < hashCodes.size(); i++) {
            map.put(hashCodes.get(i).toString(), files.get(i).split(";")[0]);
        }
        System.out.println(map.toString());

        //排序获取最大数值
        Collections.sort(hashCodes);
        String key = hashCodes.get(hashCodes.size() - 1).toString();
        System.out.println("key: " + key);

        //比较3个消息code那个数字最大,最大的就是获得锁
        System.out.println("获得分布式锁的服务IP是: " + map.get(key));

        //模拟获得锁服务处理业务
        TimeUnit.SECONDS.sleep(3);

        //处理完成业务,通知文件存储系统清理消息文件(或者文件系统的定时任务自己处理也行)
        files.clear();
        System.out.println(files.toString());

        /*
         假设一个场景,1号服务获得锁,处理业务过程中挂了,会怎么样???
         响应超时
         用户重试请求
         需要服务协调器去注册中心,根据心跳机制找到健康的服务
         然后再重新获取分布式锁
         当然这个场景不需要分布式锁,直接让服务协调器负载均衡派发请求就好了
         */
    }

    private static void writeMsg(List<String> files, List<Integer> hashCodes) throws InterruptedException {
        //计数器,用于等待线程执行完毕!
        CountDownLatch count = new CountDownLatch(3);
        new Thread(() -> {
            files.add("192.168.208.1;" + UUID.randomUUID().toString().replaceAll("-", ""));
            count.countDown();
        }).start();

        new Thread(() -> {
            files.add("192.168.208.2;" + UUID.randomUUID().toString().replaceAll("-", ""));
            count.countDown();
        }).start();

        new Thread(() -> {
            files.add("192.168.208.3;" + UUID.randomUUID().toString().replaceAll("-", ""));
            count.countDown();
        }).start();

        //等待3个线程(服务器)往文件系统写消息
        count.await();
        System.out.println("3个模拟服务器往文件存储器完毕!");
        System.out.println(files.toString());

        //将消息字符串转为正整数字
        if (files.size() > 0) {
            for (String file : files) {
                hashCodes.add(Math.abs(file.hashCode()));
            }
        }

        //重复判单,确保每个数字不相等,如果相等再来一次
        HashSet<Integer> set = new HashSet<>(hashCodes);
        if (set.size() != hashCodes.size()) {
            files.clear();
            hashCodes.clear();
            writeMsg(files, hashCodes);
        }
    }
}
相关推荐
摇滚侠8 分钟前
Spring Boot 3零基础教程,properties文件中配置和类的属性绑定,笔记14
java·spring boot·笔记
星光一影16 分钟前
HIS系统天花板,十大核心模块,门诊/住院/医保全流程打通,医院数字化转型首选
java·spring boot·后端·sql·elementui·html·scss
JAVA学习通31 分钟前
零基础OSS组件(Java)
java·linux·leetcode
草莓熊Lotso34 分钟前
《算法闯关指南:优选算法--二分查找》--19.x的平方根,20.搜索插入位置
java·开发语言·c++·算法
YJlio39 分钟前
Process Monitor 学习笔记(5.7):长时间运行追踪与日志文件体积的控制
java·笔记·学习
失散1341 分钟前
分布式专题——45 ElasticSearch基础数据管理详解
java·分布式·elasticsearch·架构
没有bug.的程序员42 分钟前
分布式监控体系:从指标采集到智能告警的完整之道
java·分布式·告警·监控体系·指标采集
想不明白的过度思考者1 小时前
JavaEE初阶——TCP/IP协议栈:从原理到实战
java·网络·网络协议·tcp/ip·java-ee
好家伙VCC1 小时前
**发散创新:渗透测试方法的深度探索与实践**随着网络安全形势日益严峻,渗透测试作为评估系统安全的
java·python·安全·web安全·系统安全
白萤2 小时前
SpringBoot用户登录注册系统设计与实现
java·spring boot·后端