实现redis分布式锁

因项目需要,需要对缓存中数据进行频繁的读写操作,为了防止并发写覆盖问题,在程序中加了redis实现的分布式锁来控制并发写的场景。

先上代码,后期更新。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
/**
 * 更新二维码中已使用次数
 *
 * @param request
 */
private void updateRechargeCodeCache(RechargeByCodeRequest request) {
    // 最大重试次数
    int retryTimes = 10;
    // 获取二维码信息
    String lockKey = "recharge_card_" + request.getChannel() + "_" + request.getRechargeCode() + "_lock";
    String rechargeCodeKey = "recharge_card_" + request.getChannel() + "_" + request.getRechargeCode();
    LoggerUtils.logBizInfo("updateRechargeCodeCache,lockKey:" + lockKey + "rechargeCodeKey:" + rechargeCodeKey);
    try {
        for (int i=0; i<retryTimes; i++) {
            // 获取分布式锁
            if (marketCacheService.setnx(lockKey, "1", 1L, TimeUnit.SECONDS)) {
                LoggerUtils.logBizInfo("updateRechargeCodeCache,lockKey:" + lockKey + ",加锁成功");
                String rechargeCodeValue = marketCacheService.get(rechargeCodeKey);
                LoggerUtils.logBizInfo("updateRechargeCodeCache,更新前:" + rechargeCodeValue);
                if (StringUtils.isEmpty(rechargeCodeValue)) {
                    throw new MarketApplicationException(MktResultCodeEnum.SCAN_CODE_INVALID);
                }
                RechargeCodeModel rechargeCodeModel = BaseDto.fromJson(rechargeCodeValue, new TypeReference<RechargeCodeModel>() {});
                // 更新二维码使用次数
                rechargeCodeModel.setUsedTimes(rechargeCodeModel.getUsedTimes() + 1);
                marketCacheService.set(rechargeCodeKey, rechargeCodeModel.toString(), 60L, TimeUnit.DAYS);
                LoggerUtils.logBizInfo("updateRechargeCodeCache,更新后:" + marketCacheService.get(rechargeCodeKey));
                // 删除分布式锁
                marketCacheService.delete(lockKey);
                LoggerUtils.logBizInfo("updateRechargeCodeCache,lockKey:" + lockKey + ",删除锁成功");
 
                break;
            } else {
                LoggerUtils.logBizInfo("updateRechargeCodeCache,lockKey:" + lockKey + ",加锁失败,重试次数:" + i);
                // 休眠10毫秒后重试
                Thread.sleep(10L);
            }
        }
    } catch (Exception e) {
        LoggerUtils.logBizError("更新二维码中已使用次数失败:" + e);
    }
}

  

 

posted on   阿泰555  阅读(114)  评论(0编辑  收藏  举报

编辑推荐:
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· 三行代码完成国际化适配,妙~啊~
· .NET Core 中如何实现缓存的预热?
· 如何调用 DeepSeek 的自然语言处理 API 接口并集成到在线客服系统
历史上的今天:
2018-01-07 [转]Jboss基础
2018-01-07 [转]HTML 简介
2018-01-07 [转]AJAX 简介
2018-01-07 [转]HTTP协议详解

导航

< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5
点击右上角即可分享
微信分享提示