分布式锁在集群的架构中发挥着重要的作用。以下有主要的使用场景

1.在秒杀、抢购等高并发场景下,多个用户同时下单同一商品,可能导致库存超卖。

2.支付、转账等金融操作需保证同一账户的资金变动是串行执行的。

3.分布式环境下,多个节点可能同时触发同一任务(如定时报表生成)。

4.用户因网络延迟重复提交表单,可能导致数据重复插入。

自定义分布式锁

获取锁

比如一下一个场景,需要对订单号为 order-88888944010的订单进行扣款处理,因为后端是多节点的,防止出现用户重复点击导致扣款请求到不用的集群节点,所以需要同时只有一个节点处理该订单。

        public static async Task<(bool Success, string LockValue)> LockAsync(string cacheKey, int timeoutSeconds = 5)
{
var lockKey = GetLockKey(cacheKey);
var lockValue = Guid.NewGuid().ToString();
var timeoutMilliseconds = timeoutSeconds * 1000;
var expiration = TimeSpan.FromMilliseconds(timeoutMilliseconds);
bool flag = await _redisDb.StringSetAsync(lockKey, lockValue, expiration, When.NotExists); return (flag, flag ? lockValue : string.Empty);
}
        public static string GetLockKey(string cacheKey)
{
return $"MyApplication:locker:{cacheKey}";
}

上述代码是在请求时将订单号作为redis key的一部分存储到redis中,并且生成了一个随机的lockValue作为值。只有当redis中不存在该key的时候才能够成功设置,即为获取到该订单的分布式锁了。


await LockAsync("order-88888944010",30); //获取锁,并且设置超时时间为30秒
释放锁
        public static async Task<bool> UnLockAsync(string cacheKey, string lockValue)
{
var lockKey = GetLockKey(cacheKey);
var script = @"local invalue = @value
local currvalue = redis.call('get',@key)
if(invalue==currvalue) then redis.call('del',@key)
return 1
else
return 0
end";
var parameters = new { key = lockKey, value = lockValue };
var prepared = LuaScript.Prepare(script);
var result = (int)await _redisDb.ScriptEvaluateAsync(prepared, parameters); return result == 1;
}

释放锁采用了lua脚本先判断lockValue是否是同一个处理节点发过来的删除请求,即判断加锁和释放锁是同一个来源。

用lua脚本而不是直接使用API执行删除的原因:

1.A获取锁后因GC停顿或网络延迟导致锁过期,此时客户端B获取了锁。若A恢复后直接调用DEL,会错误删除B持有的锁。

2.脚本在Redis中单线程执行,确保GET和DEL之间不会被其他命令打断。

自动续期

一些比较耗时的任务,可能在指定的超时时间内无法完成业务处理,需要存在自动续期的机制。

        /// <summary>
/// 自动续期
/// </summary>
/// <param name="redisDb"></param>
/// <param name="key"></param>
/// <param name="value"></param>
/// <param name="milliseconds">续期的时间</param>
/// <returns></returns>
public async static Task Delay(IDatabase redisDb, string key, string value, int milliseconds)
{
if (!AutoDelayHandler.Instance.ContainsKey(key))
return; var script = @"local val = redis.call('GET', @key)
if val==@value then
redis.call('PEXPIRE', @key, @milliseconds)
return 1
end
return 0";
object parameters = new { key, value, milliseconds };
var prepared = LuaScript.Prepare(script);
var result = await redisDb.ScriptEvaluateAsync(prepared, parameters, CommandFlags.None);
if ((int)result == 0)
{
AutoDelayHandler.Instance.CloseTask(key);
}
return;
}

保存自动续期任务的处理器

 public class AutoDelayHandler
{
private static readonly Lazy<AutoDelayHandler> lazy = new Lazy<AutoDelayHandler>(() => new AutoDelayHandler());
private static ConcurrentDictionary<string, (Task, CancellationTokenSource)> _tasks = new ConcurrentDictionary<string, (Task, CancellationTokenSource)>(); public static AutoDelayHandler Instance => lazy.Value; /// <summary>
/// 任务令牌添加到集合中
/// </summary>
/// <param name="key"></param>
/// <param name="task"></param>
/// <returns></returns>
public bool TryAdd(string key, Task task, CancellationTokenSource token)
{
if (_tasks.TryAdd(key, (task, token)))
{
task.Start(); return true;
}
else
{
return false;
}
} public void CloseTask(string key)
{
if (_tasks.ContainsKey(key))
{
if (_tasks.TryRemove(key, out (Task, CancellationTokenSource) item))
{
item.Item2?.Cancel();
item.Item1?.Dispose();
}
}
} public bool ContainsKey(string key)
{
return _tasks.ContainsKey(key);
}
}

在申请带有自动续期的分布式锁的完整代码

/// <summary>
/// 获取锁
/// </summary>
/// <param name="cacheKey"></param>
/// <param name="timeoutSeconds">超时时间</param>
/// <param name="autoDelay">是否自动续期</param>
/// <returns></returns>
public static async Task<(bool Success, string LockValue)> LockAsync(string cacheKey, int timeoutSeconds = 5, bool autoDelay = false)
{
var lockKey = GetLockKey(cacheKey);
var lockValue = Guid.NewGuid().ToString();
var timeoutMilliseconds = timeoutSeconds * 1000;
var expiration = TimeSpan.FromMilliseconds(timeoutMilliseconds);
bool flag = await _redisDb.StringSetAsync(lockKey, lockValue, expiration, When.NotExists);
if (flag && autoDelay)
{
//需要自动续期,创建后台任务
CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
var autoDelaytask = new Task(async () =>
{
while (!cancellationTokenSource.IsCancellationRequested)
{
await Task.Delay(timeoutMilliseconds / 2);
await Delay(lockKey, lockValue, timeoutMilliseconds);
}
}, cancellationTokenSource.Token);
var result = AutoDelayHandler.Instance.TryAdd(lockKey, autoDelaytask, cancellationTokenSource); if (!result)
{
autoDelaytask.Dispose();
await UnLockAsync(cacheKey, lockValue);
return (false, string.Empty);
}
}
return (flag, flag ? lockValue : string.Empty);
}

Redis的过期时间精度约为1秒,且过期检查是周期性执行的(默认每秒10次)。选择TTL/2的间隔能:

确保在Redis下一次过期检查前完成续期。

兼容Redis的主从同步延迟(通常<1秒)

StackExchange.Redis分布式锁

获取锁
string lockKey = "order:88888944010:lock";
string lockValue = Guid.NewGuid().ToString(); // 唯一标识锁持有者
TimeSpan expiry = TimeSpan.FromSeconds(10); // 锁自动过期时间
// 尝试获取锁(原子操作)
bool lockAcquired = db.LockTake(lockKey, lockValue, expiry);
释放锁
 bool released = await ReleaseLockAsync(db, lockKey, lockValue);
自动续期

同样需要自己实现

C# 使用StackExchange.Redis实现分布式锁的两种方式的更多相关文章

  1. Redis实现分布式锁的正确使用方式(java版本)

    Redis实现分布式锁的正确使用方式(java版本) 本文使用第三方开源组件Jedis实现Redis客户端,且只考虑Redis服务端单机部署的场景. 分布式锁一般有三种实现方式: 1. 数据库乐观锁: ...

  2. 分布式锁的两种实现方式(基于redis和基于zookeeper)

    先来说说什么是分布式锁,简单来说,分布式锁就是在分布式并发场景中,能够实现多节点的代码同步的一种机制.从实现角度来看,主要有两种方式:基于redis的方式和基于zookeeper的方式,下面分别简单介 ...

  3. Redis: 分布式锁的正确实现方式(转)

    前言 分布式锁一般有三种实现方式:1. 数据库乐观锁:2. 基于Redis的分布式锁:3. 基于ZooKeeper的分布式锁.本篇博客将介绍第二种方式,基于Redis实现分布式锁.虽然网上已经有各种介 ...

  4. EF+Redis(StackExchange.Redis)实现分布式锁,自测可行

    电商平台 都会有抢购的情况,比如 1元抢购. 而抢购 最重要的 就是库存,很多情况下  库存处理不好,就会出现超卖现象. 本文将用redis为缓存,StackExchange 框架,消息队列方式 实现 ...

  5. Redis实现分布式锁(设计模式应用实战)

    笔者看过网络上各种各样使用redis实现分布式锁的代码,要么错误,要么片段化,没有一个完整的例子,借这个周末给大家总结一下redis实现分布式锁的两种机制 自旋锁和排他锁 鉴于实现锁的方式不同,那么这 ...

  6. Redis的分布式锁

    一.锁的作用 当多线程执行某一业务时(特别是对数据的更新.新增)等操作,可能就会出现多个线程对同一条数据进行修改.其最终的结果一定与你期望的结果“不太一样”,这就与需要一把锁来控制线程排排队了 - j ...

  7. SpringBoot电商项目实战 — Redis实现分布式锁

    最近有小伙伴发消息说,在Springboot系列文第二篇,zookeeper是不是漏掉了?关于这个问题,其实我在写第二篇的时候已经考虑过,但基于本次系列文章是实战练习,在项目里你能看到Zookeepe ...

  8. 2020-05-24:ZK分布式锁有几种实现方式?各自的优缺点是什么?

    福哥答案2020-05-24: Zk分布式锁有两种实现方式一种比较简单,应对并发量不是很大的情况.获得锁:创建一个临时节点,比如/lock,如果成功获得锁,如果失败没获得锁,返回false释放锁:删除 ...

  9. redis持久化的两种方式RDB和AOF

    原文链接:http://www.cnblogs.com/tdws/p/5754706.html Redis的持久化过程中并不需要我们开发人员过多的参与,我们要做的是什么呢?除了深入了解RDB和AOF的 ...

  10. Redis 分布式锁,C#通过Redis实现分布式锁(转)

    目录(?)[+] 分布式锁一般有三种实现方式: 可靠性   分布式锁一般有三种实现方式: 1. 数据库乐观锁; 2. 基于Redis的分布式锁; 3. 基于ZooKeeper的分布式锁.本篇博客将介绍 ...

随机推荐

  1. 安全可信 | 首批!天翼云边缘安全加速平台AccessOne通过信通院“软件自研创新能力”专项评估

    近日,中国信息通信研究院(以下简称"中国信通院")公布"软件自研创新能力"专项评估(简称"可信研创")结果,天翼云边缘安全加速平台Acces ...

  2. Q:ssh远程连接慢的原因排查

    连接linux服务器一般都是使用SSH远程连接的方式.有时,SSH连接速度很慢,大约30s左右,但是ping时一切正常. 问题原因 1.server的sshd会去DNS查找访问的client ip的h ...

  3. [JXOI2017] 加法 题解

    最小值最大,考虑二分答案,问题转为判断最小值是否能 \(\ge x\). 假如 \(a_i\ge x\),那我们肯定不管:假如 \(a_i<x\),那最好能让选择的区间 \(r\) 值更大,用优 ...

  4. VMware虚拟机上安装Kali Linux详细教程

    1.Kali Linux简介 Kali Linux是一个基于Debian的开源Linux发行版,集成了精心挑选的渗透测试和安全审计的工具,供渗透测试和安全设计人员使用,面向各种信息安全任务:如渗透测试 ...

  5. 数据库离程序员有多远 - cnblogs救园行动感想

    这两周,我参与了博客园的"2024救园行动",成了终身会员.说实话,当初报名的时候,我心里还挺兴奋的,想着这下能和不少老朋友在这个社区里再次相聚.毕竟,在数据库行业摸爬滚打了这么多 ...

  6. 别再混淆了!JVM内存模型和Java内存模型的本质区别

    JVM 内存模型(JVM Memory Model)和 Java 内存模型(Java Memory Model, JMM)是 Java 开发中两个非常重要的概念,但这两个概念很容易被搞混,所以本文就来 ...

  7. C# 之事件及event关键字存在的意义

    总结:event关键字的作用,用于不公开发布器中委托对象实例,对事件委托对象进行保护,禁止外部调用. 1.C#事件举例说明 1 //事件及event关键字存在的意义 2 class Program 3 ...

  8. 基于PHPstream扩展手动实现一个redis客户端

    描述 redis是一个经典的key-value缓存数据库,采用C/S架构.当我们安装成功以后,你就知道它有个服务端,启动后默认监听6379端口,然后提供一个客户端工具redis-cli. 我们可以使用 ...

  9. Vue2框架-基础

    1. vue简介 什么是vue? Vue是一套用于构建用户界面的渐进式JavaScript框架.与其它大型框架不同的是,Vue 被设计为可以自底向上逐层应用.Vue 的核心库只关注视图层,方便与第三方 ...

  10. ubuntu apt 安装报错:Media change: please insert the disc labeled 'Ubuntu 20.04.5 LTS Focal Fossa - Release amd64 (20220831)' in the drive '/cdrom/' and press [Enter]

    前言 如果你在 Ubuntu 上使用 apt 安装软件包时遇到 "Media change: please insert the disc labeled ..." 的错误消息,这 ...