高并发下锁管理器,单机与分布式版

**单机版:**CommonLockManager.java

java 复制代码
package com.xxxxx.xxxxx.common.core.utils;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReentrantLock;

/**
 * 通用本地锁管理器
 * 支持:单服务并发场景
 */
public class CommonLockManager {
    // 明确使用ReentrantLock作为泛型,而非Lock接口
    private static final ConcurrentHashMap<String, ReentrantLock> LOCK_MAP = new ConcurrentHashMap<>();
    // 默认锁超时时间(3秒)
    public static final long DEFAULT_LOCK_TIMEOUT = 3;

    // 私有化构造器,禁止实例化
    private CommonLockManager() {
        throw new UnsupportedOperationException("工具类禁止实例化");
    }

    /**
     * 构建唯一锁Key:业务类型 + 业务ID(如 "job:1", "plan:2")
     */
    private static String buildLockKey(String businessType, Long businessId) {
        if (businessType == null || businessType.trim().isEmpty()) {
            throw new IllegalArgumentException("业务类型不能为空");
        }
        if (businessId == null || businessId <= 0) {
            throw new IllegalArgumentException("业务ID必须为正整数");
        }
        return businessType + ":" + businessId;
    }

    /**
     * 获取锁(不存在则自动创建)
     */
    private static ReentrantLock getLock(String lockKey) {
        // computeIfAbsent:原子创建锁,线程安全
        return LOCK_MAP.computeIfAbsent(lockKey, k -> new ReentrantLock());
    }

    /**
     * 尝试获取锁(默认3秒超时)
     * @param businessType 业务类型(如 "job", "plan", "pipe")
     * @param businessId 业务唯一ID
     * @return true=获取成功,false=超时/被占用
     */
    public static boolean tryLock(String businessType, Long businessId) {
        return tryLock(businessType, businessId, DEFAULT_LOCK_TIMEOUT);
    }

    /**
     * 尝试获取锁(自定义超时)
     * @param businessType 业务类型
     * @param businessId 业务ID
     * @param timeoutSeconds 超时时间(秒)
     * @return true=获取成功
     */
    public static boolean tryLock(String businessType, Long businessId, long timeoutSeconds) {
        String lockKey = buildLockKey(businessType, businessId);
        ReentrantLock lock = getLock(lockKey);
        try {
            // ReentrantLock原生tryLock方法,支持超时
            return lock.tryLock(timeoutSeconds, TimeUnit.SECONDS);
        } catch (InterruptedException e) {
            // 恢复线程中断状态,避免丢失
            Thread.currentThread().interrupt();
            return false;
        }
    }

    /**
     * 释放锁
     * 仅当当前线程持有该锁时,才执行释放
     */
    public static void unlock(String businessType, Long businessId) {
        String lockKey = buildLockKey(businessType, businessId);
        ReentrantLock lock = LOCK_MAP.get(lockKey);

        // 这里调用ReentrantLock的isHeldByCurrentThread()
        if (lock != null && lock.isHeldByCurrentThread()) {
            try {
                lock.unlock();
            } catch (IllegalMonitorStateException e) {
                // 防御性编程:避免未持有锁时释放导致的异常
                System.err.println("释放锁失败:当前线程未持有锁 " + lockKey);
            }
        }
    }

    /**
     * 清理未被占用的锁(避免MAP无限膨胀)
     */
    public static void cleanUnusedLocks() {
        LOCK_MAP.entrySet().removeIf(entry -> {
            ReentrantLock lock = entry.getValue();
            // 尝试获取锁,能获取到说明无线程占用,可清理
            if (lock.tryLock()) {
                lock.unlock();
                return true;
            }
            return false;
        });
    }
}

**分布式版:**CommonLockManager.java

添加Redisson依赖:

XML 复制代码
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson-spring-boot-starter</artifactId>
    <version>3.23.3</version> <!-- 选最新稳定版 -->
</dependency>
java 复制代码
// 分布式版 CommonLockManager(基于 Redisson)
@Component // 需注入 Spring 容器,因为要依赖 RedissonClient
public class CommonLockManager {
    @Resource
    private RedissonClient redissonClient;
    public static final long DEFAULT_LOCK_TIMEOUT = 3;

    private CommonLockManager() {}

    private static String buildLockKey(String businessType, Long businessId) {
        // 保持原有逻辑不变
    }

    public static boolean tryLock(String businessType, Long businessId) {
        return tryLock(businessType, businessId, DEFAULT_LOCK_TIMEOUT);
    }

    public boolean tryLock(String businessType, Long businessId, long timeoutSeconds) {
        String lockKey = buildLockKey(businessType, businessId);
        RLock lock = redissonClient.getLock(lockKey);
        try {
            return lock.tryLock(0, timeoutSeconds, TimeUnit.SECONDS); // 0秒等待,超时返回false
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            return false;
        }
    }

    public void unlock(String businessType, Long businessId) {
        String lockKey = buildLockKey(businessType, businessId);
        RLock lock = redissonClient.getLock(lockKey);
        if (lock.isHeldByCurrentThread()) {
            lock.unlock();
        }
    }
}
相关推荐
SamDeepThinking22 分钟前
你认为从0-1开发一个项目最难的地方是什么?
java·后端·架构
平安的平安23 分钟前
Python实现RAG检索增强生成:让大模型拥有你的私有知识库
开发语言·python
昵称小白25 分钟前
栈与单调栈专题
开发语言·算法
Devin~Y26 分钟前
大厂Java面试实战:Spring Boot/Cloud、Redis/Kafka、JVM调优与Spring AI RAG(内容社区UGC+AIGC客服场景)
java·jvm·spring boot·redis·spring cloud·kafka·mybatis
code bean33 分钟前
【LangChain】少样本提示(Few-Shot Prompting)实战指南
开发语言·python·langchain
青山师35 分钟前
CompletableFuture深度解析:异步编程范式与源码实现
java·单例模式·面试·性能优化·并发编程
AI人工智能+电脑小能手36 分钟前
【大白话说Java面试题 第42题】【JVM篇】第2题:JVM内存模型有哪些组成部分?
java·开发语言·jvm·面试
yqcoder36 分钟前
深入理解 JavaScript:什么是可迭代对象 (Iterable)?
开发语言·javascript·网络
破阵子4432840 分钟前
如何用 Claude Code 等 Agent 工具操作 MATLAB(支持代码编写及 Simulink)
开发语言·matlab
AI人工智能+电脑小能手43 分钟前
【大白话说Java面试题 第43题】【JVM篇】第3题:GC分为哪两种?Young GC 和 Full GC有什么区别?
java·开发语言·jvm·后端·面试