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

.NET CORE使用Redis分布式锁续命(续期)问题

结合上一期 .NET CORE 分布式事务(三) DTM实现Saga及高并发下的解决方案(.NET CORE 分布式事务(三) DTM实现Saga及高并发下的解决方案-CSDN博客)。有的小伙伴私信说如果锁内锁定的程序或者资源未在上锁时间内执行完,造成的使用资源冲突,需要如何解决。本来打算之后在发博文说明这个问题。那就先简短的说明一下。

这是一个Redis分布式锁续命或者称之为续期的问题。废话不多说,直接上代码。

using Microsoft.AspNetCore.DataProtection.KeyManagement;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
using Redlock.CSharp;
using StackExchange.Redis;
using System.Diagnostics;
using System.Globalization;
using System.Net.Sockets;
using System.Runtime.CompilerServices;
using System.Threading;public class RedisService
{private readonly ConnectionMultiplexer _redis;private readonly IDatabase _database;/// <summary>/// 初始化 <see cref="RedisService"/> 类的新实例。/// </summary>/// <param name="connectionMultiplexer">连接多路复用器。</param>public RedisService(string connectionString){_redis = ConnectionMultiplexer.Connect(connectionString);_database = _redis.GetDatabase();}#region 分布式锁#region 阻塞锁/// <summary>/// 阻塞锁--加锁/// </summary>/// <param name="key">阻塞锁的键</param>/// <param name="expireSeconds">阻塞锁的缓存时间</param>/// <param name="timeout">加锁超时时间</param>/// <returns></returns>public bool AcquireLock(string key, int expireSeconds, int timeout){var script = @"local isNX = redis.call('SETNX', KEYS[1], ARGV[1])if isNX == 1 thenredis.call('PEXPIRE', KEYS[1], ARGV[2])return 1endreturn 0";RedisKey[] scriptkey = { key };RedisValue[] scriptvalues = { key, expireSeconds * 1000 };var stopwatch = Stopwatch.StartNew();while (stopwatch.Elapsed.TotalSeconds < timeout){if (_database.ScriptEvaluate(script, scriptkey, scriptvalues).ToString() == "1"){stopwatch.Stop();return true;}}Console.WriteLine($"[{DateTime.Now}]{key}--阻塞锁超时");stopwatch.Stop();return false;}Action<string, int, int, IDatabase> postponeAction = (string key, int expireSeconds, int postponetime, IDatabase database) =>{var stopwatchpostpone = Stopwatch.StartNew();while (true){//记录时钟大于锁的设置时间说明这个锁已经自动释放了,没必要再用lua脚本去判断了,直接提前退出if (stopwatchpostpone.Elapsed.TotalSeconds > expireSeconds) return;//提前三分之一时间续命,必须提前。要不真释放了if (stopwatchpostpone.Elapsed.TotalSeconds > expireSeconds * 0.66){var scriptpostpone = @"local isNX = redis.call('EXISTS', KEYS[1])if isNX == 1 thenredis.call('PEXPIRE', KEYS[1], ARGV[2])return 1endreturn 0";RedisKey[] scriptkey = { key };RedisValue[] scriptvalues = { key, postponetime * 1000 };if (database.ScriptEvaluate(scriptpostpone, scriptkey, scriptvalues).ToString() == "1")Console.WriteLine($"[{DateTime.Now}]{key}--阻塞锁续命成功");elseConsole.WriteLine($"[{DateTime.Now}]{key}--阻塞锁续命失败");return;}}};/// <summary>/// 阻塞续命锁/// </summary>/// <param name="key">阻塞锁的键</param>/// <param name="expireSeconds">阻塞锁的缓存时间</param>/// <param name="timeout">加锁超时时间</param>/// <param name="postponetime">续命时间</param>/// <returns></returns>public bool AcquireLock(string key, int expireSeconds, int timeout, int postponetime){var script = @"local isNX = redis.call('SETNX', KEYS[1], ARGV[1])if isNX == 1 thenredis.call('PEXPIRE', KEYS[1], ARGV[2])return 1endreturn 0";RedisKey[] scriptkey = { key };RedisValue[] scriptvalues = { key, expireSeconds * 1000 };var stopwatch = Stopwatch.StartNew();while (stopwatch.Elapsed.TotalSeconds < timeout){if (_database.ScriptEvaluate(script, scriptkey, scriptvalues).ToString() == "1"){stopwatch.Stop();//锁续命Thread postponeThread = new Thread(() =>{postponeAction.Invoke(key, expireSeconds, postponetime, _database);});postponeThread.Start();return true;}}Console.WriteLine($"[{DateTime.Now}]{key}--阻塞锁超时");stopwatch.Stop();return false;}/// <summary>/// 阻塞锁--释放锁/// </summary>/// <param name="key"></param>/// <returns></returns>public bool UnAcquireLock(string key){var script = @"local getLock = redis.call('GET', KEYS[1])if getLock == ARGV[1] thenredis.call('DEL', KEYS[1])return 1endreturn 0";RedisKey[] scriptkey = { key };RedisValue[] scriptvalues = { key };return _database.ScriptEvaluate(script, scriptkey, scriptvalues).ToString() == "1";}#endregion#endregion
}

.NET CORE中是没有现成的Redis锁续命的api,只能自己造轮子。续命同样使用了Redis的Lua脚本来实现,确保了原子性。获取了Redis锁之后,直接开启了一个新的线程,在设置时间还剩三分之一的时候进行了续命,这在程序中是有必要使用的,比如说因为网络原因造成的延时,本来我的这个接口执行完毕只需要3秒钟,但是有于网络延时造成了我的这个接口执行超过了3秒,这时候就需要Redis锁续命。以上代码就可以完美结局这个问题。

    [HttpGet("AcquireLockPostpone")]public void AcquireLockPostpone(){string key = Guid.NewGuid().ToString();if (_redisService.AcquireLock("AcquireLockPostpone", 3, 100, 3)){Thread.Sleep(5000);_redisService.UnAcquireLock("AcquireLockPostpone");Console.WriteLine($"AcquireLockPostpone--释放锁");}}

控制器API,调用可以续命的阻塞锁,缓存时间设置为3秒 续命时间也是延长3秒。我们走个100阻塞锁的并发试一下。

这100个阻塞锁均续命完成。也都正常执行完毕。

相关文章:

  • java项目通用Dockerfile
  • java回溯算法笔记
  • 【OpenCV】 OpenCV (C++) 与 OpenCvSharp (C#) 之间数据通信
  • LabVIEW无人机大气数据智能测试系统
  • Unity 通过鼠标移动和LineRenderer组件实现画线功能
  • 基于Pytorch的验证码识别模型应用
  • 【python】pygame游戏框架
  • docker使用教程
  • Sentinel原理及实践
  • 图片标注编辑平台搭建系列教程(2)——fabric.js简介
  • CKS之安全沙箱运行容器:gVisor
  • JUC:线程活跃性(死锁、活锁、饥饿)
  • Python下载bing每日壁纸并实现win11 壁纸自动切换
  • MySQL详细教程
  • 从头搭建GPT(Andrej Karpathy) 笔记
  • “寒冬”下的金三银四跳槽季来了,帮你客观分析一下局面
  • 《网管员必读——网络组建》(第2版)电子课件下载
  • 345-反转字符串中的元音字母
  • Java 9 被无情抛弃,Java 8 直接升级到 Java 10!!
  • Java 内存分配及垃圾回收机制初探
  • JavaScript设计模式与开发实践系列之策略模式
  • laravel5.5 视图共享数据
  • MySQL Access denied for user 'root'@'localhost' 解决方法
  • Quartz实现数据同步 | 从0开始构建SpringCloud微服务(3)
  • react-native 安卓真机环境搭建
  • Spring核心 Bean的高级装配
  • Webpack 4x 之路 ( 四 )
  • 时间复杂度与空间复杂度分析
  • 数据科学 第 3 章 11 字符串处理
  • 消息队列系列二(IOT中消息队列的应用)
  • 做一名精致的JavaScripter 01:JavaScript简介
  • Oracle Portal 11g Diagnostics using Remote Diagnostic Agent (RDA) [ID 1059805.
  • 分布式关系型数据库服务 DRDS 支持显示的 Prepare 及逻辑库锁功能等多项能力 ...
  • 曾刷新两项世界纪录,腾讯优图人脸检测算法 DSFD 正式开源 ...
  • # 日期待t_最值得等的SUV奥迪Q9:空间比MPV还大,或搭4.0T,香
  • ## 临床数据 两两比较 加显著性boxplot加显著性
  • #git 撤消对文件的更改
  • #经典论文 异质山坡的物理模型 2 有效导水率
  • #我与Java虚拟机的故事#连载05:Java虚拟机的修炼之道
  • #我与Java虚拟机的故事#连载16:打开Java世界大门的钥匙
  • $.proxy和$.extend
  • (a /b)*c的值
  • (LeetCode C++)盛最多水的容器
  • (NSDate) 时间 (time )比较
  • (附源码)计算机毕业设计SSM疫情下的学生出入管理系统
  • (六)vue-router+UI组件库
  • (免费领源码)python#django#mysql校园校园宿舍管理系统84831-计算机毕业设计项目选题推荐
  • (免费领源码)python+django+mysql线上兼职平台系统83320-计算机毕业设计项目选题推荐
  • (十一)图像的罗伯特梯度锐化
  • (原創) 如何優化ThinkPad X61開機速度? (NB) (ThinkPad) (X61) (OS) (Windows)
  • .form文件_SSM框架文件上传篇
  • .mkp勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复
  • .NET Standard、.NET Framework 、.NET Core三者的关系与区别?
  • .net连接oracle数据库
  • .pings勒索病毒的威胁:如何应对.pings勒索病毒的突袭?