C#如何实现分布式锁 Redis分布式锁C#实现方法

C#中实现Redis分布式锁需用SET命令原子加锁并设唯一value,通过Lua脚本安全解锁防误删,配合Watchdog机制自动续期,再辅以重连、重试、日志等生产级保障。

c#如何实现分布式锁 redis分布式锁c#实现方法

在C#中实现Redis分布式锁,核心是利用Redis的SET命令原子性设置带过期时间的key,并配合唯一value(如GUID)做锁标识,再通过Lua脚本保证解锁操作的原子性。关键不在“加锁”,而在“防误删”和“自动续期”——否则容易出现锁失效或锁被别人释放的问题。

用StackExchange.Redis实现基础分布式锁

推荐使用StackExchange.Redis客户端,它性能高、线程安全、支持连接池。加锁本质是执行:

SET lock:key "unique-token" NX PX 30000:NX表示key不存在才设,PX设30秒过期 返回”OK”表示加锁成功;nil表示已存在,加锁失败 value必须全局唯一(建议用Guid.NewGuid().ToString()),避免A加锁后B误删

安全解锁必须用Lua脚本

不能先GET再DEL!否则可能刚查到key属于你,下一刻就过期,接着DEL删掉别人刚抢到的锁。正确做法是用Lua保证“判断+删除”原子执行:

Lua脚本示例:
if redis.call("get", KEYS[1]) == ARGV[1] then return redis.call("del", KEYS[1]) else return 0 end C#中调用:db.ScriptEvaluate(luaScript, new RedisKey[] { key }, new RedisValue[] { token }) 返回1表示解锁成功,0表示token不匹配(锁已被他人持有或已过期)

进阶:支持自动续期(Watchdog机制)

业务执行时间不确定?别把锁过期时间设太长(影响故障恢复),也别设太短(怕没执行完就释放)。推荐用后台定时任务自动续期:

加锁时启动一个Timer,每隔expireTime / 3(如10秒)尝试执行GETSET key new-token并比对旧value 仅当当前value仍为自己的token时,才更新过期时间(可用EXPIRE命令) 任务在业务完成或异常时及时取消,避免“幽灵续期”

生产环境建议补充项

真正上线前,别只盯着加解锁逻辑:

使用连接串中的abortConnect=false和重连策略,避免Redis短暂不可用导致锁异常 加锁失败时,建议用指数退避重试(如100ms→200ms→400ms),而非死循环 记录加锁/解锁日志(含key、token、耗时),便于排查“锁未释放”或“重复释放”问题 考虑封装成IDistributedLock接口,方便后续替换为ZooKeeper或数据库方案

基本上就这些。不复杂但容易忽略细节——尤其是value唯一性和Lua解锁。用对了,Redis分布式锁在C#里很稳。

以上就是C#如何实现分布式锁 Redis分布式锁C#实现方法的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1443010.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 19:37:32
下一篇 2025年12月10日 06:26:12

相关推荐

  • C#怎么实现单例模式 C#设计模式之单例实现方法

    推荐使用Lazy实现单例模式,线程安全且支持延迟初始化;其次可选静态构造函数方式(非延迟);DCL仅用于旧框架兼容。需注意避免滥用、优先考虑DI容器替代。 在C#中实现单例模式,核心是确保一个类只有一个实例,并提供全局访问点。最常用、最推荐的是静态构造函数 + 私有静态只读字段方式(线程安全、简洁、…

    2025年12月17日
    000
  • Blazor 弹窗(Modal)怎么实现

    Blazor纯C#实现Modal弹窗无需JS,核心是状态驱动显隐、防滚动、点击遮罩/ESC关闭、焦点管理及可配置样式;通过RenderFragment支持嵌套内容,配合CSS遮罩与动画,兼顾可访问性与体验。 Blazor 实现弹窗(Modal)不依赖 JS,纯 C# + Razor 就能搞定,核心是…

    2025年12月17日 好文分享
    000
  • C#怎么获取当前路径 C#获取程序运行目录方法

    最常用且安全的方式是使用AppDom%ignore_a_1%n.CurrentDomain.BaseDirectory获取exe所在目录;Environment.CurrentDirectory返回当前工作目录但可能变化;跨平台推荐AppContext.BaseDirectory或Assembly.…

    2025年12月17日
    000
  • C#怎么创建抽象类 C# abstract class与virtual方法

    抽象类用abstract声明,不可实例化,只能继承;可含已实现成员和必须由子类重写的abstract成员;含abstract成员的类必须声明为abstract;子类须override所有abstract成员,除非自身也声明为abstract。 在C#中,抽象类用 abstract 关键字声明,不能被…

    2025年12月17日
    000
  • C# 如何使用MemoryStream – 在内存中进行流操作

    MemoryStream 是 C# 中基于内存的流实现,继承自 Stream,用于临时存储、序列化、加密等场景,可提升性能;需注意内存占用、位置重置和及时释放资源。 在 C# 中,MemoryStream 是一种基于内存的流实现,允许你在不涉及磁盘或网络的情况下对数据进行读写操作。它继承自 Stre…

    2025年12月17日
    000
  • C# 信号量(Semaphore)的应用 – 控制对资源的并发访问数

    应使用 SemaphoreSlim 控制并发许可数而非线程数,适用于限流场景;需用 try/finally 或 C#12+ using 确保 Release,避免许可泄露。 信号量(Semaphore)在 C# 中是用来限制同时访问某资源的线程数量的同步原语。它不像 lock 那样只允许一个线程进入…

    2025年12月17日
    000
  • Blazor 创建和注入 API Service 的方法

    Blazor中创建和注入API Service的核心是定义接口与实现类封装HTTP调用,通过DI注册(WebAssembly用Singleton、Server用Scoped),并在OnInitializedAsync中异步调用;需注意HttpClient配置、CORS、生命周期匹配及错误处理。 在 …

    2025年12月17日
    000
  • C# yield关键字的作用 – 实现迭代器与状态机的简便方法

    yield关键字用于声明迭代器方法,使方法能逐个提供序列元素并自动管理状态;返回类型须为IEnumerable等,编译器自动生成状态机,支持延迟计算与内存优化。 yield 关键字在 C# 中不是用来“返回值”或“跳出方法”的,而是专门用于声明迭代器方法(iterator method)——它让方法…

    2025年12月17日
    000
  • C#怎么遍历枚举 C# Enum.GetValues遍历方法

    在C#中遍历枚举最常用方式是Enum.GetValues,返回Array类型,需显式转换为具体枚举类型或使用C# 7.3+泛型重载;配合GetNames可获取名称与值;Flags枚举需手动位运算生成组合值。 在 C# 中遍历枚举最常用、最直接的方式就是用 Enum.GetValues,它返回一个包含…

    2025年12月17日
    000
  • Blazor 怎么自定义表单验证消息

    Blazor自定义表单验证消息主要有三种方式:一是用DataAnnotations+自定义ValidationAttribute实现字段级规则;二是用EditContext.ValidationMessageStore动态添加运行时业务错误;三是手动监听EditContext事件实现复杂联动校验。 …

    2025年12月17日
    000
  • Blazor 怎么实现拖放功能

    Blazor拖放依赖HTML5 drag & drop API,需设draggable=”true”、用@ondragstart/@ondragover/@ondrop绑定事件,注意阻止默认行为、数据传递及跨平台限制。 Blazor 实现拖放功能主要靠 HTML5 原…

    2025年12月17日
    000
  • Blazor WASM AOT 提升运行时性能的方法

    AOT编译可提升Blazor WebAssembly性能,但需显式启用、精简反射依赖、优化渲染逻辑并结合Web API协同调优。 Blazor WebAssembly(WASM)启用 AOT(Ahead-of-Time)编译后,能显著减少 JIT 开销、缩短启动时间、提升执行效率。但 AOT 本身不…

    2025年12月17日
    000
  • C# For和Foreach循环的性能差异 – 遍历集合的最佳选择

    for在数组和List等支持随机访问的集合中通常更快,因无枚举器开销;foreach更安全可读,适合多数场景且不易出错。 在C#中,for 和 foreach 都能遍历集合,但性能表现并不总是一样。关键看集合类型、是否需要索引、以及是否在循环中修改集合 —— 这些因素直接影响哪种写法更高效、更安全。…

    2025年12月17日
    000
  • C# init访问器是什么 – C# 9.0中的只读属性初始化

    init访问器是C# 9.0引入的仅限对象初始化阶段使用的属性设置器,支持在对象初始化器中为只读属性赋值,之后不可修改,增强不可变性,适用于DTO、record等场景。 init访问器是 C# 9.0 引入的一种特殊属性设置器,用于在对象初始化期间为只读属性赋值,之后该属性便不能再被修改。它解决了以…

    2025年12月17日
    000
  • ASP.NET Core怎么进行身份验证和授权 JWT认证实现教程

    ASP.NET Core 中 JWT 身份验证核心是配置 Authentication 与 Authorization 中间件,流程为登录发 Token → 请求带 Token → 中间件自动校验 → 控制器用 [Authorize] 限定访问;需正确注册 JWT Bearer 服务、生成 Toke…

    2025年12月17日
    000
  • Blazor 怎么给 HTTP 请求添加认证头

    Blazor中HTTP认证头配置核心是按场景选择方式:WASM推荐用DelegatingHandler动态注入Bearer Token,Server端通常无需前端加头。 在 Blazor 中给 HTTP 请求添加认证头,核心是配置 HttpClient 实例,使其自动携带如 Authorizatio…

    2025年12月17日
    000
  • C#如何实现PDF文件生成 QuestPDF库使用教程

    QuestPDF 是 .NET 平台最轻量现代的 PDF 生成库,基于 Fluent API 用 C# 直接描述文档结构,需注册中文字体(如 Noto Sans CJK)以支持中文,安装 via NuGet,.NET 6+ 原生兼容。 用 C# 生成 PDF,QuestPDF 是目前最轻量、现代且易…

    2025年12月17日
    000
  • C#如何使用dynamic关键字 C#动态类型dynamic用法

    dynamic是C#中将类型检查推迟到运行时的特殊类型,适用于COM交互、反射调用、动态JSON处理及DSL构建等场景,但会丢失编译检查与IDE支持。 dynamic 是 C# 中用于绕过编译时类型检查的特殊类型,它把类型解析推迟到运行时。用它不是为了“替代”静态类型,而是为了解决某些特定场景下类型…

    2025年12月17日
    000
  • C# 中的Monitor.Wait和Pulse – 底层的线程同步原语

    Monitor.Wait 和 Monitor.Pulse 是 C# 中基于对象的线程同步原语,用于实现等待-通知机制:Wait 释放锁并使线程进入等待队列,Pulse 向队列中一个线程发送唤醒信号但不释放锁,二者需配合循环条件检查使用。 在 C# 中,Monitor.Wait 和 Monitor.P…

    2025年12月17日
    000
  • WPF怎么实现数据绑定 WPF MVVM数据绑定方法

    WPF数据绑定依赖DependencyProperty和INotifyPropertyChanged,ViewModel需实现该接口并触发PropertyChanged事件,View通过DataContext关联ViewModel,Binding路径须为public属性且区分大小写,集合应使用Obs…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信