.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-03
ASP.NET?Core設(shè)置Ocelot網(wǎng)關(guān)限流
這篇文章介紹了ASP.NET?Core設(shè)置Ocelot網(wǎng)關(guān)限流的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-04-04
基于.Net中的數(shù)字與日期格式化規(guī)則助記詞的使用詳解
本篇文章是對.Net中的數(shù)字與日期格式化規(guī)則助記詞的使用進行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
在 asp.net core 的中間件中返回具體的頁面的實現(xiàn)方法
這篇文章主要介紹了在 asp.net core 的中間件中返回具體的頁面的實現(xiàn)方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-08-08

