Redisson 分布式锁的最佳实践
Redisson 分布式锁的最佳实践
- 第一、添加依赖
- 第二、添加redisson配置类
- 第三、添加测试类
- 测试结果
- 扩展知识
- redisson锁中lock方法和tryLock方法有什么区别
- 锁续约
- 注意事项
引言
在现代分布式系统中,处理并发问题是至关重要的。分布式锁是解决这类问题的关键工具之一。本文将介绍如何使用 Redisson,一个强大的 Redis 客户端库,来实现高性能、高可用性的分布式锁。
基本概念
分布式锁是多个进程或线程之间协调操作的一种机制。它通常包括三个基本操作:加锁、持有锁、释放锁。Redisson 提供了简单而强大的接口来管理这些操作。
Redisson 分布式锁的使用方法
第一、添加依赖
<!-- redisson分布式锁 -->
<dependency><groupId>org.redisson</groupId><artifactId>redisson</artifactId><version>3.15.5</version>
</dependency>
第二、添加redisson配置类
package com.kingmouse.lock;import org.redisson.Redisson;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;/*** Created with IntelliJ IDEA** @author kingMouse* @date 2023/11/7 09:50* Description: redisson配置类 单机和集群自选*/@Configuration
public class RedissonConfig {/*** 单机模式* @return RedissonClient*/@Bean(destroyMethod = "shutdown")public RedissonClient redissonSingle() {// 1.创建配置Config config = new Config();// 单机模式config.useSingleServer()// 设置地址.setAddress("redis://127.0.0.1:6379")// 设置连接数.setConnectionPoolSize(10);return Redisson.create(config);}/*** 单机模式* @return RedissonClient*/// @Bean(destroyMethod = "shutdown")
// public RedissonClient redissonCluster() {
// // 1.创建配置
// Config config = new Config();
// // 集群模式-带密码
// config.useClusterServers()
// .addNodeAddress("127.0.0.1:7004", "127.0.0.1:7001")
// .setPassword("12356")
// // 设置master节点最小连接数
// .setMasterConnectionMinimumIdleSize(10)
// // 设置slave节点最小连接数
// .setSlaveConnectionMinimumIdleSize(10);
// return Redisson.create(config);
// }
第三、添加测试类
package com.kingmouse.lock;import lombok.extern.slf4j.Slf4j;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;import java.util.concurrent.TimeUnit;/*** Created with IntelliJ IDEA** @author kingMouse* @date 2023/11/5 09:54* Description:*/@Slf4j
@RunWith(SpringRunner.class)
@SpringBootTest
public class RedissonLockTest {@Autowiredprivate RedissonClient redissonClient;/*** 阻塞式*/@Testpublic void testRedissonBlock() {// 设置锁的名字,同一个名字为同一把锁,不同名字之间互不干扰String lockKey = "kingmouse-001";// 获取锁对象RLock lock = redissonClient.getLock(lockKey);// 加锁,尝试获取锁,5秒后获取不到则直接放弃lock.lock(5, TimeUnit.SECONDS);// 实际业务处理try {System.out.println("阻塞式--加锁成功,线程 ID:" + Thread.currentThread().getId());} catch (Exception e) {log.error("业务处理异常:{}", e.getMessage());} finally {// 解锁lock.unlock();System.out.println("阻塞式--解锁成功,线程 ID:" + Thread.currentThread().getId());}}@Testpublic void testRedissonNonBlock() throws Exception {// 设置锁的名字,同一个名字为同一把锁,不同名字之间互不干扰String lockKey = "kingmouse-002";// 获取锁对象RLock lock = redissonClient.getLock(lockKey);// 尝试获取锁,最长等待5秒,持有锁最长30秒boolean isLock = lock.tryLock(5, 30, TimeUnit.SECONDS);if (isLock) {// 实际业务处理try {System.out.println("非阻塞式--获取锁成功,线程 ID:" + Thread.currentThread().getId());} catch (Exception e) {log.error("业务处理异常:{}", e.getMessage());} finally {// 解锁lock.unlock();System.out.println("非阻塞式--解锁成功,线程 ID:" + Thread.currentThread().getId());}}}}
测试结果
阻塞式--加锁成功,线程 ID:1
阻塞式--解锁成功,线程 ID:1
非阻塞式--获取锁成功,线程 ID:1
非阻塞式--解锁成功,线程 ID:1
扩展知识
redisson锁中lock方法和tryLock方法有什么区别
在 Redisson 中,lock 方法和 tryLock 方法都是用来获取分布式锁的,但它们在获取锁的方式和行为上有一些区别:
在 Redisson 中,lock 方法和 tryLock 方法都是用来获取分布式锁的,但它们在获取锁的方式和行为上有一些区别:
-
lock方法:lock方法是阻塞的,即如果获取锁失败,调用该方法的线程会被阻塞,直到获取到锁为止。- 如果在获取锁时发生网络分区或其他异常,会一直重试获取锁,直到获取成功或者达到设置的超时时间为止。
- 使用
lock方法时,如果获取锁成功后,需要手动调用unlock方法来释放锁。
例如:
RLock lock = redisson.getLock("myLock");lock.lock();try {// 执行业务逻辑} finally {lock.unlock();}
-
tryLock方法:tryLock方法是非阻塞的,即如果获取锁失败,方法会立即返回,不会阻塞当前线程。tryLock方法可以设置超时时间,在指定的超时时间内尝试获取锁,如果超时仍未获取到锁,则返回false。- 使用
tryLock方法时,获取锁成功后不需要手动释放锁,因为锁会在超时时间后自动释放。(建议使用完自动释放锁)
例如:
RLock lock = redisson.getLock("myLock");boolean isLocked = lock.tryLock(10, TimeUnit.SECONDS);if (isLocked) {try {// 执行业务逻辑} finally {lock.unlock();}} else {// 获取锁失败的处理逻辑}
总结来说,lock 方法是阻塞的,会一直尝试获取锁直到成功,而 tryLock 方法是非阻塞的,在指定的超时时间内尝试获取锁,如果超时则返回失败。选择使用哪种方法取决于你的业务需求,以及是否能够容忍等待获取锁的阻塞时间。
锁续约
使用看门狗(Watchdog)可以实现锁的自动续约:
String permitId = lock.acquire(10, TimeUnit.SECONDS);
RScheduledExecutorService executorService = redisson.getExecutorService("myExecutorService");
executorService.schedulePermitExpiration(permitId, 5, TimeUnit.SECONDS);
在这个例子中,我们使用 schedulePermitExpiration 方法每隔5秒续约一次锁的持有时间。
注意事项
在使用分布式锁时,需要注意以下几点:
-
合理设置锁的持有时间:不要将锁的持有时间设置得过长,以免影响系统的并发性能。根据实际业务需求,选择一个合适的持有时间。
-
使用看门狗进行续约:在需要锁的持有时间较长的情况下,使用看门狗可以确保锁不会在持有时间内过期,避免了手动续约的繁琐操作。
-
异常处理:在加锁、续约、释放锁的过程中,需要考虑可能出现的异常情况,并进行适当的处理,以确保锁能够被正确释放,避免死锁等问题。
实际应用场景
在电商系统中,分布式锁可以用来避免商品超卖问题。在任务调度系统中,可以确保同一个任务在同一时刻只被一个节点执行,避免任务重复执行等问题。
总结
Redisson 提供了简单而强大的分布式锁解决方案,通过合理设置锁的持有时间和使用看门狗进行续约,可以确保系统在高并发环境下的稳定性和可靠性。在实际应用中,开发人员可以根据业务需求选择合适的锁策略,并进行必要的异常处理,以实现分布式系统的高效运行。
相关文章:
Redisson 分布式锁的最佳实践
Redisson 分布式锁的最佳实践 第一、添加依赖第二、添加redisson配置类第三、添加测试类测试结果扩展知识redisson锁中lock方法和tryLock方法有什么区别锁续约 注意事项 引言 在现代分布式系统中,处理并发问题是至关重要的。分布式锁是解决这类问题的关键工具之一。…...
ArkTS声明式开发范式
装饰器 用来装饰类、结构体、方法以及变量,赋予其特殊的含义,如上述示例中 Entry 、 Component 、 State 都是装饰器。 Component 表示这是个自定义组件; Entry 则表示这是个入口组件; State 表示组件中的状态变量,…...
史诗级云故障敲响警钟,应用保障不能没有“连续键”!
近日,知名云服务商出现一次史诗级的云故障:全球所有区域/所有服务同时异常,故障持续长达3小时之多,云上众多应用受到极大影响。 如今,在一个充满不确定性和复杂性的数字化时代,哪怕是顶级云服务商亦不能避…...
SSH连接远程服务器报错:WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED 解决方法
一.错误描述 报错信息里提示了路径信息/root/.ssh/known_hosts:20 二.解决方案 方法一 输入以下指令: ssh-keygen -R XXX(需要连接远程服务器的ip) 按照我的例子ip:10.165.7.136,会返回以下信息: 重新尝试连接: 输…...
数据库——查询连续的月份
一、GP或PGSQL with recursive t(n) as (select date(2023-01-01) union all select n1 from t where n < now()) select to_char(n, yyyy-mm) as ny from t group by ny order by ny 二、Hive select add_months(FROM_UNIXTIME(unix_timestamp(SUBSTR(start_date, 1, 7…...
git代码提交命令(如何提交代码)
# 提交暂存区到仓库区 $ git commit -m [message]# 提交暂存区的指定文件到仓库区 $ git commit [file1] [file2] ... -m [message]# 提交工作区自上次commit之后的变化,直接到仓库区 $ git commit -a# 提交时显示所有diff信息 $ git commit -v# 使用一次新的commit…...
jmeter中调用python代码
1、安装pyinstaller pip install -i https://pypi.tuna.tsinghua.edu.cn/simple pyinstaller 2、将py脚本打包 pyinstaller -F venv/get_image/OCR_jmeter_api.py 3、jmeter中添加OS Process Sampler并调用dist下的程序 4、执行jmeter...
当当网获得dangdang商品详情商品列表API 测试请求入口
item_get-获得dangdang商品详情 获取商品详情 item_search-按关键字搜索dangdang商品 获取商品列表 公共参数 名称类型必须描述keyString是调用key(必须以GET方式拼接在URL中)secretString是调用密钥api_nameString是API接口名称(包括在请…...
git如何查看配置,修改配置,设置配置
# 显示当前的Git配置 $ git config --list# 编辑Git配置文件 $ git config -e [--global]# 设置提交代码时的用户信息 $ git config [--global] user.name "[name]" $ git config [--global] user.email "[email address]"...
交通流合成数据生成原理及实现代码
移动数据是设备的地理位置,通过正常活动被动产生。 它具有从交通规划到迁移预测的重要应用。 由于移动数据稀有且难以收集,研究人员已开始探索综合生成移动数据的解决方案。 在本文中,我将讨论一种用于合成移动数据的简单解决方案。 该合成数…...
leetcode 240. 搜索二维矩阵 II
2023.11.22 本题最先想到的是暴力法和二分法,暴力法就不写了,写一下二分法的解法,java代码如下: class Solution {public boolean searchMatrix(int[][] matrix, int target) {for(int[] row : matrix){int left 0;int right r…...
a标签超链接 —— 实现点击前中后变色
浅浅记录下 <style type"text/css"> a:link {color: yellow; /*未访问链接颜色*/ }a:visited {color: red; /*已访问链接颜色*/ }a:hover {color: blue; /*鼠标移动到链接颜色*/text-decoration: underline; }a:active {color: orange; /*鼠标点击时颜色*/ }a…...
【好玩的开源项目】Linux系统之部署proxx扫清黑洞小游戏
【好玩的开源项目】Linux系统之部署proxx扫清黑洞小游戏 一、proxx小游戏介绍1.1 proxx小游戏简介1.2 开源地址 二、本地环境介绍2.1 本地环境规划2.2 本次实践介绍 三、检查本地环境3.1 检查本地操作系统版本3.2 检查系统内核版本 四、部署Node.js环境4.1 下载Node.js安装包4.…...
IDEA-SVN合并分支到主干
IDEA-SVN合并branch分支到主干master 1.选择VCS的 Integrate Project 2.选择分支合并 Source1 是合并后的分支 , 主分支 master Source2 是被合并的分支 , 分支 branch Try merge 可以尝试是否可以能够被合并,并且无冲突 3.合并完成后当前项目会出现需要提交的内容,检查一…...
kettle如何写日志
var subject"自定义日志输出"; //实例化工厂类 var logFactory new org.pentaho.di.core.logging.LogChannelFactory(); //实例化日志channel对象 var log logFactory.create(subject); //日志输出 log.logMinimal("XXXXXXXXXXXXXXXXXXXXXXXX-preRows"acc…...
新能源车将突破2000万辆,汉威科技为电池安全保驾护航
近年来,我国新能源汽车销量持续突破新高。据中汽协数据,1~10月,国内新能源汽车销量达728万辆,同比增长37.8%,市场占有率达到30.4%。随着第四季度车市传统旺季的到来,新能源消费需求将进一步释放,…...
基于文心一言AI大模型,编写一段python3程序以获取华为分布式块存储REST接口的实时数据
本文尝试基于文心一言AI大模型,编写一段python3程序以获取华为分布式块存储REST接口的实时数据。 一、用文心一言AI大模型将需求转化为样例代码 1、第一次对话:“python3写一段从rest服务器获取数据的样例代码” 同时生成了以下注解 这段代码首先定义…...
2022-4-11 南科大现代控制与最优估计
CLEAR_LAB B站视频 矩阵的分块矩阵操作 diagonal 对角阵 identity matrix 单位矩阵 矩阵克罗内克积...
【注册Huggingface】获取token
Hugging Face是一家美国公司,专门开发用于构建机器学习应用的工具。该公司的代表产品是其为自然语言处理应用构建的transformers库,以及允许用户共享机器学习模型和数据集的平台。 Huggingface 是一个开源的cv、nlp框架,提供了超过100,000个…...
【蓝桥杯软件赛 零基础备赛20周】第4周——简单模拟1
文章目录 什么是简单模拟简单模拟和编程能力刷题 什么是简单模拟 正在学编程语言(C/C、Python、Java),或者刚学过语言,还没有开始学数据结构和算法的同学,有一些疑问:如何快速入门算法竞赛?如何…...
【超详细】Allan偏差+PSD八大可视化一文吃透:随机游走频率噪声从原理到画图全流程(附公式与工程避坑)
文章目录一、为什么要“多视角可视化”理解随机游走频率噪声1. 单一图形判断误区2. 工程现实:长时稳定性才是系统“生死线”3. 本文解决什么问题二、随机游走频率噪声的本质(用直觉彻底搞懂)1. 数学定义:频率的“积分噪声”模型2.…...
Swoole v5.1.3 + LLM推理服务长连接架构(附可运行架构图+Docker Compose+性能基线报告)
更多请点击: https://intelliparadigm.com 第一章:Swoole v5.1.3 LLM推理服务长连接架构概览 Swoole v5.1.3 作为 PHP 领域领先的协程化网络引擎,其对 WebSocket、HTTP/2 和自定义 TCP 协议的原生支持,为构建低延迟、高并发的 …...
DAComp:大语言模型多维评估基准与工程实践
1. 项目背景与核心价值DAComp作为新一代大语言模型评估基准,正在重新定义AI测试方法论。这个由数据科学家和AI工程师共同打造的开源工具,解决了当前LLM评估中的三大痛点:评估维度单一、测试场景脱离实际、缺乏全流程追踪。我在实际参与多个LL…...
Python京东茅台抢购终极指南:毫秒级精准定时自动化脚本
Python京东茅台抢购终极指南:毫秒级精准定时自动化脚本 【免费下载链接】jd_maotai 抢京东茅台脚本,定时自动触发,自动预约,自动停止 项目地址: https://gitcode.com/gh_mirrors/jd/jd_maotai 在电商秒杀活动中,…...
Switchyard:基于Python的用户空间网络仿真与协议测试实践指南
1. 项目概述:一个面向网络仿真与测试的“数字沙盘”如果你和我一样,长期混迹在网络开发、协议研究或者网络安全测试的圈子里,那你一定对“网络仿真”这个词不陌生。无论是想验证一个新路由算法的收敛速度,还是想模拟一个复杂的跨数…...
强化学习感知的知识蒸馏框架RLAD解析
1. 强化学习感知的知识蒸馏框架解析在大型语言模型(LLM)的推理能力优化领域,知识蒸馏(Knowledge Distillation)与强化学习(Reinforcement Learning)的结合正成为突破模型性能瓶颈的关键路径。传统蒸馏方法在静态监督微调(SFT)场景表现良好,但当遇到强化学…...
Qwen3大模型推理优化与注意力机制实践
1. 项目背景与核心价值Qwen3作为当前开源大模型领域的重要代表,其技术架构的演进方向直接影响着行业应用落地的可能性。这份技术报告最吸引我的地方在于它没有停留在常规的精度对比层面,而是深入剖析了模型规模与注意力机制这两个决定推理成本的关键维度…...
HiClaw 1.1.0:企业级 Agent 开发的基建升级
我最近在做一个企业 AI 培训项目,帮客户部署智能体平台。说实话,技术能力早就不是问题,真正的挑战是怎么让它在各种奇葩环境里稳稳当当跑起来。 上周刚交付一个项目,用的是 1.0.9 版本。客户验收那天说"挺稳的"&#x…...
C++实现动态绑定代码分享
C实现动态绑定代码分享 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 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 #include…...
深入UDS 0x23服务:从内存映射到安全访问,搞懂汽车ECU数据读取的那些‘坑’
深入UDS 0x23服务:从内存映射到安全访问,搞懂汽车ECU数据读取的那些‘坑’ 当你在深夜的办公室里调试一台报错的ECU,突然发现某个关键参数异常,而唯一能验证猜想的方式就是直接读取内存数据——这时0x23服务(ReadMemor…...
