当前位置: 首页 > news >正文

redis分布式秒杀锁

在这里插入图片描述

-- 获取锁标识,是否与当前线程一致?
if(redis.call('get', KEYS[1]) == ARGV[1]) then-- 一致,删除return redis.call('del', KEYS[1])
end
-- 不一致,直接返回
return 0

在这里插入图片描述

package com.platform.lock;public interface ILock {/*** 获取锁* @param timeoutSec* @return*/public boolean tryLock(long timeoutSec);/*** 锁标识、释放锁*/public void unlock();
}

在这里插入图片描述

package com.platform.lock;import cn.hutool.core.lang.UUID;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.io.ClassPathResource;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.stereotype.Service;import java.util.Collections;
import java.util.concurrent.TimeUnit;/*** redis的分布式锁* 实现ILock接口*/
public class SimpleRedisLock implements ILock {// 不同的业务有不同的锁名称private String name;private StringRedisTemplate stringRedisTemplate;private static final String KEY_PREFIX = "tryLock:";private static final String ID_PREFIX = UUID.randomUUID().toString(true) + "-";// DefaultRedisScript,private static final DefaultRedisScript<Long> UNLOCK_SCRIPT;public SimpleRedisLock(String name, StringRedisTemplate stringRedisTemplate) {this.name = name;this.stringRedisTemplate = stringRedisTemplate;}// 初始化 UNLOCK_SCRIPT,用静态代码块的方式,一加载SimpleRedisLock有会加载unlock.lua// 避免每次调unLock() 才去加载,提升性能!!!static {UNLOCK_SCRIPT = new DefaultRedisScript<>();// setLocation() 设置脚本位置UNLOCK_SCRIPT.setLocation(new ClassPathResource("unlock.lua"));// 返回值类型UNLOCK_SCRIPT.setResultType(Long.class);}/*** 获取锁*/@Overridepublic boolean tryLock(long timeoutSec) {// 获取线程标示String threadId = ID_PREFIX + Thread.currentThread().getId();// 获取锁// set lock thread1 nx ex 10// nx : setIfAbsent(如果不存在) , ex : timeoutSec(秒)Boolean success = stringRedisTemplate.opsForValue().setIfAbsent(KEY_PREFIX + name, threadId, timeoutSec, TimeUnit.SECONDS);// 自动拆箱(Boolean -> boolean)!!!可能有风险return Boolean.TRUE.equals(success);}/*** 解决判断(锁标识、释放锁)这两个动作,之间产生阻塞!!!* JVM的 FULL GC* 要让这两个动作具有原子性*/@Overridepublic void unlock() {// 调用lua脚本stringRedisTemplate.execute(UNLOCK_SCRIPT,Collections.singletonList(KEY_PREFIX + name),ID_PREFIX + Thread.currentThread().getId());}
}

在这里插入图片描述

    @PostMapping("/cancelPublicBenefit")@CheckRepeatCommitpublic RestResponse cancelPublicBenefit(@LoginUser MallUserEntity loginUser, @RequestBody MallPublicBenefitEntity publicBenefitEntity) {String key = PUBLIC_BENEFIT_TEAM_LOCK_FLAG + publicBenefitEntity.getId();RestResponse restResponse = null;SimpleRedisLock simpleRedisLock = new SimpleRedisLock(key,stringRedisTemplate);try {if (simpleRedisLock.tryLock(15)) {// 成功获取锁,执行业务逻辑restResponse = mallPublicBenefitService.cancelPublicBenefit(loginUser, publicBenefitEntity);} else {// 获取锁失败,处理失败逻辑throw new BusinessException("服务器繁忙!");}} finally {simpleRedisLock.unlock();}return restResponse;}

在这里插入图片描述

http://www.lryc.cn/news/186144.html

相关文章:

  • 【Redis】String内部编码方式
  • 川西旅游网系统-前后端分离(前台vue 后台element UI,后端servlet)
  • Paddle使用pyinstaller打包出错的解决方法
  • 【Java acm】特殊输入
  • 在Ubuntu 20.04搭建最小实验环境
  • 使用uwsgi部署Flask
  • Android平台实现lottie动画
  • JAVA练习百题之求矩阵对角线之和
  • MEM备考打卡
  • 短视频矩阵源码开发部署---技术解析
  • 百度小程序制作源码 百度引流做关键词排名之技巧
  • 【计算机组成 课程笔记】7.3 高速缓存 Cache
  • vscode搭建c/c++环境
  • macOS Sonoma 14.0(23A344) 正式版带 OpenCore 0.9.6 和 FirPE 三分区镜像
  • 神经网络(MLP多层感知器)
  • git与github的交互(文件与文件夹的上传)
  • Visual Studio常见编译错误记录
  • 如何应对数据安全四大挑战?亚马逊云科技打出“组合拳”
  • JavaScript——数据类型、类型转换
  • C位操作符
  • 【linux进程(三)】进程有哪些状态?--Linux下常见的三种进程状态
  • numString.charAt(i) - ‘0‘
  • 《Python 自动化办公应用大全》书籍推荐(包邮送书五本)
  • day57:ARMday4,程序状态寄存器读写指令、软中断指令、C和汇编的混合编程、开发板介绍
  • el-cascader
  • 图论第3天----第841题、第463题
  • 软件测试/测试开发丨利用ChatGPT 生成自动化测试脚本
  • 3.3.OpenCV技能树--二值图像处理--图像形态学操作
  • 这15个海运提单的雷区 你知道吗?
  • 几道web题目