.NET?Core使用Redis實現(xiàn)創(chuàng)建分布式鎖
在 .NET Core WebApi 中使用 Redis 創(chuàng)建分布式鎖可以通過 StackExchange.Redis
庫來實現(xiàn)。分布式鎖用于確保在分布式系統(tǒng)中,同一時間只有一個進程可以執(zhí)行某段代碼。
1. 場景描述
在支付系統(tǒng)中,可能會出現(xiàn)以下并發(fā)問題:
- 用戶同時發(fā)起多次支付請求,導(dǎo)致重復(fù)扣款。
- 多個請求同時處理同一個訂單,導(dǎo)致數(shù)據(jù)不一致。
通過分布式鎖,可以確保同一時間只有一個請求能夠執(zhí)行關(guān)鍵操作(如扣款)。
2. 實現(xiàn)步驟
2.1 安裝 StackExchange.Redis 包
首先,安裝 Redis 客戶端庫:
dotnet add package StackExchange.Redis
2.2 配置 Redis 連接
在 appsettings.json
中添加 Redis 連接字符串:
{ "ConnectionStrings": { "Redis": "localhost:6379" } }
2.3 創(chuàng)建分布式鎖工具類
創(chuàng)建一個工具類來封裝 Redis 分布式鎖的邏輯:
using StackExchange.Redis; using System; using System.Threading.Tasks; public class RedisDistributedLock { private readonly IDatabase _redisDatabase; private readonly string _lockKey; private readonly string _lockValue; private readonly TimeSpan _expiry; public RedisDistributedLock(IDatabase redisDatabase, string lockKey, string lockValue, TimeSpan expiry) { _redisDatabase = redisDatabase; _lockKey = lockKey; _lockValue = lockValue; _expiry = expiry; } public async Task<bool> AcquireLockAsync() { // 嘗試設(shè)置鎖,僅當(dāng)鍵不存在時才成功 return await _redisDatabase.StringSetAsync(_lockKey, _lockValue, _expiry, When.NotExists); } public async Task ReleaseLockAsync() { // 使用 Lua 腳本確保只有鎖的持有者才能釋放鎖 var luaScript = @" if redis.call('GET', KEYS[1]) == ARGV[1] then return redis.call('DEL', KEYS[1]) else return 0 end"; await _redisDatabase.ScriptEvaluateAsync(luaScript, new RedisKey[] { _lockKey }, new RedisValue[] { _lockValue }); } }
2.4 在 Web API 中使用分布式鎖
在 Web API 的控制器中使用分布式鎖來確保支付操作的原子性。
2.4.1 注冊 Redis 服務(wù)
在 Startup.cs
或 Program.cs
中注冊 Redis 服務(wù):
// 添加 Redis 服務(wù) builder.Services.AddSingleton<IConnectionMultiplexer>(sp => ConnectionMultiplexer.Connect(builder.Configuration.GetConnectionString("Redis")));
2.4.2 創(chuàng)建支付控制器
在 Controllers
文件夾中創(chuàng)建一個 PaymentController
,并在其中使用分布式鎖:
using Microsoft.AspNetCore.Mvc; using StackExchange.Redis; using System; using System.Threading.Tasks; [ApiController] [Route("api/[controller]")] public class PaymentController : ControllerBase { private readonly IDatabase _redisDatabase; public PaymentController(IConnectionMultiplexer redis) { _redisDatabase = redis.GetDatabase(); } [HttpPost("pay")] public async Task<IActionResult> ProcessPayment([FromBody] PaymentRequest request) { // 創(chuàng)建分布式鎖 var lockKey = $"PaymentLock:{request.OrderId}"; // 鎖的鍵,基于訂單 ID var lockValue = Guid.NewGuid().ToString(); // 鎖的值,確保唯一性 var expiry = TimeSpan.FromSeconds(10); // 鎖的過期時間 var distributedLock = new RedisDistributedLock(_redisDatabase, lockKey, lockValue, expiry); try { // 嘗試獲取鎖 if (await distributedLock.AcquireLockAsync()) { Console.WriteLine("已獲取鎖,正在處理付款..."); // 模擬支付處理 bool paymentSuccess = await ProcessPaymentAsync(request.UserId, request.OrderId, request.Amount); if (paymentSuccess) { return Ok(new { Message = "付款成功!" }); } else { return BadRequest(new { Message = "付款失敗!" }); } } else { return Conflict(new { Message = "正在處理此訂單的另一個付款請求..." }); } } finally { // 釋放鎖 await distributedLock.ReleaseLockAsync(); } } }
3. 代碼說明
3.1 分布式鎖的實現(xiàn)
AcquireLockAsync
: 使用 Redis
的 SET key value NX EX
命令嘗試獲取鎖。NX
表示僅在鍵不存在時設(shè)置,`EX 設(shè)置鍵的過期時間。
ReleaseLockAsync
: 使用 Lua
腳本確保只有鎖的持有者才能釋放鎖,避免誤刪其他請求的鎖。
3.2 支付控制器的使用
鎖的鍵: 使用訂單 ID 作為鎖的鍵(如 PaymentLock:202501061410455506968463210
),確保同一訂單的支付請求串行化。
鎖的值: 使用 GUID 作為鎖的值,確保鎖的唯一性。
鎖的過期時間: 設(shè)置合理的過期時間(如 10 秒),防止鎖被長時間占用。
3.3 支付處理邏輯
ProcessPaymentAsync
: 模擬支付處理邏輯,包括調(diào)用支付網(wǎng)關(guān)、扣減余額等操作。
4. 測試 API
4.1 啟動 Web API
運行項目,啟動 Web API。
4.2 發(fā)送支付請求
使用工具(如 Postman 或 curl)發(fā)送支付請求:
POST /api/payment/pay Content-Type: application/json { "userId": "9527", "orderId": "202501061410455506968463210" }
4.3 測試并發(fā)場景
同時發(fā)送多個相同的支付請求,觀察是否只有一個請求能夠成功獲取鎖并處理支付。
5. 注意事項
鎖的粒度:
- 鎖的粒度要適中。如果鎖的粒度過大(如全局鎖),可能導(dǎo)致性能問題;如果粒度過小,可能增加復(fù)雜性。
- 在支付系統(tǒng)中,通常以訂單 ID 或用戶 ID 作為鎖的粒度。
鎖的過期時間:
- 設(shè)置合理的過期時間,避免鎖被長時間占用導(dǎo)致死鎖。
- 如果業(yè)務(wù)邏輯執(zhí)行時間較長,可以動態(tài)延長鎖的過期時間。
鎖的可靠性:
Redis 需要高可用,否則可能導(dǎo)致鎖失效。可以使用 Redis 集群或 Redlock 算法提高可靠性。
異常處理:
確保鎖的釋放操作放在 finally
塊中,避免因異常導(dǎo)致鎖無法釋放。
冪等性:
支付系統(tǒng)需要支持冪等性,即使多次請求,也只會產(chǎn)生一次扣款。
6. 總結(jié)
在 .NET Core Web API 中使用 Redis 創(chuàng)建分布式鎖,可以帶來以下好處:
- 解決并發(fā)問題,確保數(shù)據(jù)一致性。
- 提高系統(tǒng)的可靠性和性能。
- 簡化代碼邏輯,降低開發(fā)復(fù)雜度。
- 支持高并發(fā)、分布式環(huán)境和高可用需求。
通過合理使用 Redis 分布式鎖,可以構(gòu)建高可靠、高性能的分布式系統(tǒng),滿足復(fù)雜的業(yè)務(wù)需求。
到此這篇關(guān)于.NET Core使用Redis實現(xiàn)創(chuàng)建分布式鎖的文章就介紹到這了,更多相關(guān).NET Redis創(chuàng)建分布式鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Community Server專題三:HttpModule
Community Server專題三:HttpModule...2007-03-03ASP.NET?Core設(shè)置Ocelot網(wǎng)關(guān)限流
這篇文章介紹了ASP.NET?Core設(shè)置Ocelot網(wǎng)關(guān)限流的方法,文中通過示例代碼介紹的非常詳細。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-04-04基于.Net中的數(shù)字與日期格式化規(guī)則助記詞的使用詳解
本篇文章是對.Net中的數(shù)字與日期格式化規(guī)則助記詞的使用進行了詳細的分析介紹,需要的朋友參考下2013-05-05在 asp.net core 的中間件中返回具體的頁面的實現(xiàn)方法
這篇文章主要介紹了在 asp.net core 的中間件中返回具體的頁面的實現(xiàn)方法,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-08-08