SpringBoot集成Lock4j 底层使用Redission 实现分布锁

news/2024/12/22 19:53:07/

Lock4j

在这里插入图片描述

在分布式系统中,实现锁的功能对于保证数据一致性和避免并发冲突是非常重要的。Lock4j是一个简单易用的分布式锁框架,而Redisson是一个功能强大的分布式解决方案,可以与Lock4j进行集成。

操作步骤

第一步:添加依赖

首先,在你的Spring Boot项目的pom.xml文件中添加以下依赖:

<dependencies><!--若使用redisTemplate作为分布式锁底层,则需要引入--><dependency><groupId>com.baomidou</groupId><artifactId>lock4j-redis-template-spring-boot-starter</artifactId><version>${latest.version}</version></dependency><!--若使用redisson作为分布式锁底层,则需要引入--><dependency><groupId>com.baomidou</groupId><artifactId>lock4j-redisson-spring-boot-starter</artifactId><version>${latest.version}</version></dependency><!--若使用zookeeper作为分布式锁底层,则需要引入--><dependency><groupId>com.baomidou</groupId><artifactId>lock4j-zookeeper-spring-boot-starter</artifactId><version>${latest.version}</version></dependency>
</dependencies>

这些依赖将会引入Redisson和Lock4j以及它们的必要依赖。

第二步:配置Redis

在src/main/resources目录下创建application.yml文件,并添加以下配置:

spring:redis:host: 127.0.0.1port: 6379password: ddz123database: 0

这是一个示例配置文件,你需要根据自己的实际情况修改Redis的连接信息,如地址、端口、密码等。

第三步:使用@Lock4j注解实现分布式锁

在你的Spring Boot应用程序中,创建一个测试类,用于ApiPost的调用两次第一个会获取到锁,第二个获取锁失败会抛出异常。

    @GetMapping("/lock")@Lock4j(keys = {"#id"}, acquireTimeout = 100, expire = 10000
//            , executor = RedissonLockExecutor.class)public String getLock(String id) throws InterruptedException {Thread.sleep(10000);return id;}

在这里插入图片描述

自定义锁

自定义锁key的生成规则

只需要实现LockKeyBuilder接口中 buildKey() 方法即可定义自己的生成规则。

@Component
public class CustomKeyBuilder implements LockKeyBuilder {/*** 构建key** @param invocation     invocation* @param definitionKeys 定义* @return key*/@Overridepublic String buildKey(MethodInvocation invocation, String[] definitionKeys) {// 自定义锁key的生成策略return null;}
}

自定义锁失败策略

实现LockFailureStrategy接口中onLockFailure() 方法即可自定义获取锁失败返回策略。

@Component
public class GrabLockFailureStrategy implements LockFailureStrategy {/*** 锁失败事件** @param key       锁key* @param method    方法* @param arguments 自变量*/@Overridepublic void onLockFailure(String key, Method method, Object[] arguments) {throw new RuntimeException("获取锁失败了!");}
}

自定义加锁解锁逻辑

@Component
public class RedissonLockExecutor extends AbstractLockExecutor<RLock> {/*** 加锁** @param lockKey        锁标识* @param lockValue      锁值* @param expire         锁有效时间* @param acquireTimeout 获取锁超时时间* @return 锁信息*/@Overridepublic RLock acquire(String lockKey, String lockValue, long expire, long acquireTimeout) {return null;}/*** 解锁** <pre>* 为何解锁需要校验lockValue* 客户端A加锁,一段时间之后客户端A解锁,在执行releaseLock之前,锁突然过期了。* 此时客户端B尝试加锁成功,然后客户端A再执行releaseLock方法,则将客户端B的锁给解除了。* </pre>** @param key          加锁key* @param value        加锁value* @param lockInstance 锁实例* @return 是否释放成功*/@Overridepublic boolean releaseLock(String key, String value, RLock lockInstance) {return false;}
}

注意事项

在使用Spring Boot集成Lock4j和Redisson时,需要注意以下事项:

  • 确保Redisson的配置正确。在redisson.yaml文件(或application.properties)中,你需要提供正确的Redis连接信息,如地址、端口、密码等。确保你的Redis服务运行正常。
  • 注意锁的名称要保证唯一性。不同的锁需要使用不同的锁名称。使用相同名称的锁可能会导致不同方法之间的锁冲突。
  • 考虑对方法异常进行处理。为了确保锁能够被正确释放,你可以在方法中使用try-catch-finally结构,并在finally块中释放锁。这样即使方法出现异常,也能保证锁的守护不会因异常而中断。

通过以上步骤和注意事项,你可以轻松地在Spring Boot项目中集成Lock4j,并使用Redisson作为底层实现分布式锁。这样,你可以安全地处理分布式系统中的并发问题,并保证数据的一致性。


http://www.ppmy.cn/news/987251.html

相关文章

​《吐血整理》进阶系列教程-拿捏Fiddler抓包教程(9)-Fiddler如何设置捕获Https会话​

1.简介 由于近几年来各大网站越来越注重安全性都改成了https协议&#xff0c;不像前十几年前直接是http协议直接裸奔在互联网。还有的小伙伴或者童鞋们按照上一篇宏哥的配置都配置好了&#xff0c;想大展身手抓一下百度的包&#xff0c;结果一试傻眼了&#xff0c;竟然毛都没有…

小程序相较于APP,广告变现有哪些优势?

对于开发者而言&#xff0c;微信小程序开发门槛相对较低&#xff0c;难度不及APP&#xff0c;能够满足简单的基础应用&#xff0c;适合生活服务类线下商铺以及非刚需低频应用的转换。 和早期相比&#xff0c;今天小程序在产品功能、UI设计、交互体验等方面&#xff0c;越来越精…

基金经理二季度AI概念股操作分化

公募基金二季度仍在加仓AI板块&#xff0c;但不同于一季度全线加仓题材各环节&#xff0c;二季度对AI产业链的操作出现分化。 资金更加聚拢在业绩率先兑现的上游算力板块。其中光模块、服务器是加仓最为显著的两个领域&#xff1b;对于部分业绩短期兑现前景不明的AI板块&#…

差分数组工具类

差分数组工具类 什么是差分数组工具类 什么是差分数组 其实差分数组和前缀和数组有点相似 前缀和主要适用于原始数组不会被修改的情况下&#xff0c;需要频繁查询某个区间的累计和 preSum[i]就是nums[0…i-1]所有元素的累加和 若求nums[i…j]的累加和&#xff0c;则计算preSum…

2023牛客多校第三场 B.Auspiciousness

传送门 前题提要:没得说,赛时根本没想到dp,赛后翻各大题解看了很久,终于懂了dp的做法,故准备写一篇题解. 首先,先定义一下我们的 d p dp dp方程,考虑将处于 [ 1 , n ] [1,n] [1,n]的数当做小数,将处于 [ n 1 , 2 ∗ n ] [n1,2*n] [n1,2∗n]的数当做大数.那么对于我们的摸牌结…

【SA8295P 源码分析】55 - ifs2_la.img 镜像加载解析过程分析

【SA8295P 源码分析】55 - ifs2_la.img 镜像加载解析过程分析 一、startupmgr 可执行程序分析1. startupmgr\src\script.c 入口 main 函数:调用 init_loader_and_launcher 解析 scripts 数组二、ifsloader镜像加载流程分析:2.1 init_loader_and_launcher() :初始化rlimit rl…

学C的第三十一天(上)【通讯录的实现】

相关代码gitee自取&#xff1a;C语言学习日记: 加油努力 (gitee.com) 接上期&#xff1a; 学C的第三十天【自定义类型&#xff1a;结构体、枚举、联合】_高高的胖子的博客-CSDN博客 通讯录需求&#xff1a; 实现一个通讯录&#xff0c; 通讯录中存放保存人的信息&#xff1…

信号完整性分析基本概念之PRBS码型

在信号完整性测试或者仿真过程中&#xff0c;我们通常需要给一个通道发送PRBS码型&#xff0c;那么什么是PRBS码型呢&#xff1f; PRBS全称是Pseudo-Random Binary Sequence&#xff0c;翻译过来就是伪随机二进制序列&#xff0c;什么是伪随机呢&#xff0c;随机就是说PRBS码型…