windows下使用redis解决.net6.0下人工调用接口时分配位置的并发问题
使用了nuget包包括了:CSRedisCore,StackExchange.Redis,MyStack.DistributedLocking,Microsoft.Extensions.Configuration
安装Redis并注册为windows服务 直接参考这位兄弟的成果:
https://www.cnblogs.com/qingheshiguang/p/17952623
注册服务:
data:image/s3,"s3://crabby-images/6da44/6da44a3c422e49abcf1dae786223d28e774e2de6" alt=""
配置文件appsetting.json中加上Redis的连接字符串 "Redis": "127.0.0.1:6379", Program.cs中添加注册服务 // 添加并发锁支持 builder.Services.AddDistributedLock(builder.Configuration["Redis"]);
data:image/s3,"s3://crabby-images/6da44/6da44a3c422e49abcf1dae786223d28e774e2de6" alt=""
using CSRedis; using Microsoft.Extensions.Caching.Distributed; using Microsoft.Extensions.Caching.Redis; using Utility.Concurrent; namespace Microsoft.Extensions.DependencyInjection { public static class ServiceCollectionExtensions { /// <summary> /// 添加并发锁支持 /// </summary> /// <param name="services"></param> /// <param name="connectionString"></param> /// <returns></returns> public static IServiceCollection AddDistributedLock(this IServiceCollection services, string connectionString) { RedisHelper.Initialization(new CSRedisClient(connectionString)); services.AddSingleton<IDistributedCache>(new CSRedisCache(RedisHelper.Instance)); services.AddSingleton(typeof(IDistributedLock), typeof(DistributedLock)); return services; } } }
redis客户端实现
data:image/s3,"s3://crabby-images/6da44/6da44a3c422e49abcf1dae786223d28e774e2de6" alt=""
using CSRedis; namespace Concurrent { public class DistributedLock : IDistributedLock { private const string _lockKey = "LockKey"; public async Task<bool> TryLockAsync(string key, int lockExpirySeconds = 10, long waitLockSeconds = 60) { if (string.IsNullOrEmpty(key)) { throw new ArgumentNullException("锁键值不能为空。"); } DateTime begin = DateTime.Now; while (true) { if (await RedisHelper.SetAsync($"{_lockKey}.{key}", Thread.CurrentThread.ManagedThreadId, lockExpirySeconds, RedisExistence.Nx)) { return true; } //不等待锁则返回 if (waitLockSeconds == 0) { break; } //超过等待时间,则不再等待 if ((DateTime.Now - begin).TotalSeconds >= waitLockSeconds) { break; } Thread.Sleep(100); } return false; } public async Task UnLockAsync(string key) { await RedisHelper.DelAsync($"{_lockKey}.{key}"); } public async Task<T> LockExecuteAsync<T>(string key, Func<Task<T>> handle, int lockExpirySeconds = 10, long waitLockSeconds = 60) { if (await TryLockAsync(key, lockExpirySeconds, waitLockSeconds)) { try { return await handle?.Invoke(); } finally { try { await UnLockAsync(key); } finally { } } } return await Task.FromResult(default(T)); } } }
//锁定分配的目标库位 var hasLock = await _distributedLock.TryLockAsync(LocationCode, 5, 10); if (!hasLock) { throw new BizException(_Localizer["操作频繁,稍后再试"]); }
业务中调用
本文来自博客园,作者:十四,转载请注明原文链接:https://www.cnblogs.com/yanghucheng/p/18208318
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· [翻译] 为什么 Tracebit 用 C# 开发
· Deepseek官网太卡,教你白嫖阿里云的Deepseek-R1满血版
· DeepSeek崛起:程序员“饭碗”被抢,还是职业进化新起点?
· 2分钟学会 DeepSeek API,竟然比官方更好用!
· .NET 使用 DeepSeek R1 开发智能 AI 客户端