Redisson分布式锁实战

使用场景

工作中遇到的场景:在多节点部署的微服务项目中,某一条规则可能正在调度,正在清理,或正在导入。这三种操作都会操作数据库,且没有开启事务。为了保证数据一致性,它们不能同时发生。

所以需要实现多个服务在同一时间对于同一资源的互斥访问

解决方案

在redis中存储每个规则的状态processStatus,1代表正在调度,2代表正在清理,3代表正在导入。

每次在调度,清理,导入之前,从redis中获取processStatus。

如果有值,说明正在操作,直接根据processStatus的值返回给用户相应提示。

如果没有值,说明该规则处于空闲状态。

        1. 为该规则的processStatus设置对应的值(这个操作在高并发的时候不安全,需要分布式锁)

        2. 开始进行相应操作

        3. 操作结束的finally代码块中,删除redis中processStatus这个key

代码实现

使用redission提供的分布式锁,因为它内置了锁对象自动续期,自旋等待,防止死锁等机制

@Slf4j
    public class RedisDistributedLockUtil {

        private RedissonClient redissonClient = SpringContextHolder.getBean(RedissonClient.class);
        private String lockKey;
        private String status; // 存放处理标记数据
        private boolean locked = false; // 记录是否上锁成功

        public RedisDistributedLockUtil(String key, String status) {
            this.lockKey = key;
            this.status = status;
        }

        public void setAndCheckFlagWithLock() {
            RBucket<Object> bucket = redissonClient.getBucket("quality:" + lockKey + ":isProcessing");
            // first check
            checkBucket(bucket);
            RLock rLock = redissonClient.getLock("quality:" + lockKey + ":lock");
            try {
                rLock.lock();
                // double check
                checkBucket(bucket);
                bucket.set(status, 300000L, TimeUnit.MILLISECONDS);
                locked = true;
                log.info("{}添加redis导入标志位", lockKey);
            } finally {
                rLock.unlock();
            }
        }

        private static void checkBucket(RBucket<Object> bucket) {
            if (CollectionUtils.isNotEmpty(bucket.get())) {
                String str = String.format("该笔规则正在%s,请稍后重试。", RuleProcessingState.getDescByCode(String.valueOf(bucket.get())));
                throw new ServiceException(1, str);
            }
        }

        public void deleteFlag() {
            if (!locked) {
                return;
            }

            try {
                RBucket<Object> bucket = redissonClient.getBucket("quality:" + lockKey + ":isProcessing");
                bucket.delete();
                log.info("删除redis导入标志位 {}", lockKey);
            } catch (Exception e) {
                e.printStackTrace();
                log.error("清理redis标志位失败! {} [{}]", lockKey, e.getMessage());
                throw new ServiceException(1, e.getMessage());
            }
        }
    }

在相应操作的时候使用分布式锁,以Excel导入为例

@Override
public void doAfterAllAnalysed(AnalysisContext analysisContext) {
    try {
        if (CollectionUtils.isEmpty(processDetailEntities) &&
            CollectionUtils.isEmpty(processEntities) &&
            CollectionUtils.isEmpty(resultEntities)) {
            redisDistributedLockUtil.deleteFlag();  // 确保直接return时锁被释放
            return;
        }

        if (CollectionUtils.isNotEmpty(errMsg)) {
            redisDistributedLockUtil.deleteFlag();  // 确保直接return时锁被释放
            return;
        }

        checkOrgCode(); // 检验获取机构代码信息
        if (CollectionUtils.isNotEmpty(errMsg)) {
            redisDistributedLockUtil.deleteFlag();  // 确保直接return时锁被释放
            return;
        }

        orgCode2rowIndex.clear();

        // 提交异步任务,负责执行导入和释放锁
        globalThreadPool.submit(() -> {
            try {
                // 初始化日志
                logUtil.initLog(ruleEntity);

                // 执行导入操作
                startImport();
            } catch (Exception e) {
                log.error("{} 初始化或导入过程中出错:{}", ruleCode, e.getMessage());
                e.printStackTrace();
                errMsg.add(String.format("操作失败!请检查:%s", e.getMessage()));
            } finally {
                // 确保异步线程结束时释放锁
                redisDistributedLockUtil.deleteFlag();
            }
        });

    } catch (Exception e) {
        log.error("主线程执行过程中发生异常:{}", e.getMessage());
        errMsg.add(String.format("操作失败!请检查:%s", e.getMessage()));
        redisDistributedLockUtil.deleteFlag();  // 确保异常时释放锁
    }
}


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值