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

spring-integration-redis中分布式锁RedisLockRegistry的使用

pom依赖:

<!-- redis -->

<dependency>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-starter-data-redis</artifactId>

</dependency>

<dependency>

<groupId>org.springframework.integration</groupId>

<artifactId>spring-integration-redis</artifactId>

</dependency>

配置:

@Configuration

public class RedisLockConfig {

@Bean

public RedisLockRegistry redisLockRegistry(RedisConnectionFactory redisConnectionFactory) {

//第一个参数redisConnectionFactory

//第二个参数registryKey,分布式锁前缀,设置为项目名称会好些

//该构造方法对应的分布式锁,默认有效期是60秒.可以自定义

return new RedisLockRegistry(redisConnectionFactory, "boot-launch");

//return new RedisLockRegistry(redisConnectionFactory, "boot-launch",60);

}

}

方式1:

Lock lock = redisLockRegistry.obtain("lockkey-" + 1);
//tryLock尝试获取锁(成功返回true,失败返回false);if (lock.tryLock()) {try {// 业务代码块} finally {//走到这里说明获取锁一定是成功的try{lock.unlock();}catch (IllegalStateException e){//业务操作时间过长,会导致锁超时内系统释放,这里就会产生超时异常log.error("释放时发现锁过期");}}} else {//走到这里说明获取锁一定是失败的的log.info("并发抢锁失败");}

方式2:

Lock lock = redisLockRegistry.obtain("lockkey-" + 1);
//lock()一直循环获取锁,循环周期为100毫秒(成功了则无返回,失败了则抛异常),理论上一定能获取到锁,因为会一直循环取锁try {lock.lock();// 业务代码块(这里要判断并发情况下的业务幂等性)}catch (CannotAcquireLockException e){//当redis挂掉时执行redis操作失败会产生异常CannotAcquireLockException;说明获取锁失败了log.error("get redis lock fail");} finally {//走到这里 获取锁不一定是成功的,异常时也走,因此需要判断lock状态if(Objects.noNull(lock)){//锁存在才需要释放try{lock.unlock();}catch (IllegalStateException e){//业务操作时间过长,会导致锁超时内系统释放,这里就会产生超时异常log.error("释放时发现锁过期");}}}

方式3:

Lock lock = redisLockRegistry.obtain("lockkey-" + 1);
try {
//tryLock(3, TimeUnit.SECONDS):在3秒钟内循环获取锁,超过3秒还没有获取到就认为获取失败了,这个方法会抛出异常InterruptedException,boolean isLocked = lock.tryLock(3, TimeUnit.SECONDS);if (isLocked) {// 业务代码块}} catch (InterruptedException e) {log.error("get redis lock fail");}finally {//获取锁失败的时候也会走,需要判断锁失败时不去释放if(Objects.noNull(lock)){//锁存在才需要释放try{lock.unlock();}catch (IllegalStateException e){//业务操作时间过长,会导致锁超时内系统释放,这里就会产生超时异常log.error("释放时发现锁过期");}}}

总结:

解决同一个用户并发操作同一个资源时,如提交时按钮连击、多端调用同一个业务接口:使用方式1;方式2(使用方式2时需要做幂等判断,如连击操作多次调用时,对第二次调用不做处理)

解决并发限流时,如抽奖系统中对商品sku库存做锁操作,防止多个用户同时扣减库存导致库存数量少扣:使用方式2、或者方式3(锁等待时间=预计业务代码执行时间)

三方式都要注意释放锁时做锁过期处理

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

相关文章:

  • 城市通电(prim算法)
  • 【动态规划】
  • 秒懂算法 | DP概述和常见DP面试题
  • 【C++提高编程】C++全栈体系(二十五)
  • 【云原生】k8s核心技术—集群安全机制 Ingress Helm 持久化存储-20230222
  • 【Linux】实现简易的Shell命令行解释器
  • 再获认可!腾讯安全NDR获Forrester权威推荐
  • 代码审计之旅之百家CMS
  • ONLYOFFICE中利用chatGPT帮助我们策划一场生日派对
  • Java面试题-线程(一)
  • 一篇普通的bug日志——bug的尽头是next吗?
  • Vue 3 第八章:Watch侦听器
  • GlassFish的安装与使用
  • 【java】Java 重写(Override)与重载(Overload)
  • OpenCV-PyQT项目实战(12)项目案例08:多线程视频播放
  • 面向对象设计模式:结构型模式之装饰器模式
  • Unity iOS 无服务器做一个排行榜 GameCenter
  • 现在招个会自动化测试的人是真难呀~你会个锤子的自动化测试
  • OracleDatabase——数据库表空间dmp导出与导入
  • 20张图带你彻底了解ReentrantLock加锁解锁的原理
  • Dockerfile构建Springboot镜像
  • 从深分页查询到覆盖索引
  • Go语言学习的第三天--下部分(Gin框架的基础了解)
  • JDK的动态代理(powernode 文档)(内含源代码)
  • 第1章 多线程基础
  • Linux基本指令(一)
  • el-dialog子组件在mounted周期内获取不到dom?
  • 第九章 opengl之光照(光照贴图)
  • JDK动态代理(powernode CD2207 video)(内含教学视频+源代码)
  • 【Linux】Sudo的隐晦bug引发的一次业务问题排查