欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C#使用StackExchange.Redis實現(xiàn)分布式鎖的兩種方式介紹

 更新時間:2025年04月27日 16:23:59   作者:BruceNeter  
分布式鎖在集群的架構(gòu)中發(fā)揮著重要的作用,這篇文章主要為大家介紹了C#使用StackExchange.Redis實現(xiàn)分布式鎖的兩種方式,希望對大家有一定的幫助

分布式鎖在集群的架構(gòu)中發(fā)揮著重要的作用。以下有主要的使用場景

1.在秒殺、搶購等高并發(fā)場景下,多個用戶同時下單同一商品,可能導(dǎo)致庫存超賣。

2.支付、轉(zhuǎn)賬等金融操作需保證同一賬戶的資金變動是串行執(zhí)行的。

3.分布式環(huán)境下,多個節(jié)點可能同時觸發(fā)同一任務(wù)(如定時報表生成)。

4.用戶因網(wǎng)絡(luò)延遲重復(fù)提交表單,可能導(dǎo)致數(shù)據(jù)重復(fù)插入。

自定義分布式鎖

獲取鎖

比如一下一個場景,需要對訂單號為 order-88888944010的訂單進行扣款處理,因為后端是多節(jié)點的,防止出現(xiàn)用戶重復(fù)點擊導(dǎo)致扣款請求到不用的集群節(jié)點,所以需要同時只有一個節(jié)點處理該訂單。

        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作為值。只有當(dāng)redis中不存在該key的時候才能夠成功設(shè)置,即為獲取到該訂單的分布式鎖了。

            await LockAsync("order-88888944010",30); //獲取鎖,并且設(shè)置超時時間為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腳本先判斷l(xiāng)ockValue是否是同一個處理節(jié)點發(fā)過來的刪除請求,即判斷加鎖和釋放鎖是同一個來源。

用lua腳本而不是直接使用API執(zhí)行刪除的原因:

1.A獲取鎖后因GC停頓或網(wǎng)絡(luò)延遲導(dǎo)致鎖過期,此時客戶端B獲取了鎖。若A恢復(fù)后直接調(diào)用DEL,會錯誤刪除B持有的鎖。

2.腳本在Redis中單線程執(zhí)行,確保GET和DEL之間不會被其他命令打斷。

自動續(xù)期

一些比較耗時的任務(wù),可能在指定的超時時間內(nèi)無法完成業(yè)務(wù)處理,需要存在自動續(xù)期的機制。

        /// <summary>
        /// 自動續(xù)期
        /// </summary>
        /// <param name="redisDb"></param>
        /// <param name="key"></param>
        /// <param name="value"></param>
        /// <param name="milliseconds">續(xù)期的時間</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;
        }

保存自動續(xù)期任務(wù)的處理器

 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>
     /// 任務(wù)令牌添加到集合中
     /// </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);
     }
 }

在申請帶有自動續(xù)期的分布式鎖的完整代碼

/// <summary>
/// 獲取鎖
/// </summary>
/// <param name="cacheKey"></param>
/// <param name="timeoutSeconds">超時時間</param>
/// <param name="autoDelay">是否自動續(xù)期</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)
    {
        //需要自動續(xù)期,創(chuàng)建后臺任務(wù)
        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秒,且過期檢查是周期性執(zhí)行的(默認每秒10次)。選擇TTL/2的間隔能:

確保在Redis下一次過期檢查前完成續(xù)期。

兼容Redis的主從同步延遲(通常<1秒)

StackExchange.Redis分布式鎖

獲取鎖

string lockKey = "order:88888944010:lock";
string lockValue = Guid.NewGuid().ToString(); // 唯一標(biāo)識鎖持有者
TimeSpan expiry = TimeSpan.FromSeconds(10);   // 鎖自動過期時間
// 嘗試獲取鎖(原子操作)
bool lockAcquired = db.LockTake(lockKey, lockValue, expiry);

釋放鎖

 bool released = await ReleaseLockAsync(db, lockKey, lockValue);

自動續(xù)期

同樣需要自己實現(xiàn)

到此這篇關(guān)于C#使用StackExchange.Redis實現(xiàn)分布式鎖的兩種方式介紹的文章就介紹到這了,更多相關(guān)C# StackExchange.Redis實現(xiàn)分布式鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C#中Html.RenderPartial與Html.RenderAction的區(qū)別分析

    C#中Html.RenderPartial與Html.RenderAction的區(qū)別分析

    這篇文章主要介紹了C#中Html.RenderPartial與Html.RenderAction的區(qū)別分析,需要的朋友可以參考下
    2014-07-07
  • C#?webApi創(chuàng)建與發(fā)布、部署、api調(diào)用詳細教程

    C#?webApi創(chuàng)建與發(fā)布、部署、api調(diào)用詳細教程

    這篇文章主要給大家介紹了關(guān)于C#?webApi創(chuàng)建與發(fā)布、部署、api調(diào)用的相關(guān)資料,WebApi是微軟在VS2012?MVC4版本中綁定發(fā)行的,WebApi是完全基于Restful標(biāo)準的框架,文中通過圖文介紹的非常詳細,需要的朋友可以參考下
    2023-12-12
  • C#畫筆Pen繪制自定義線的帽子

    C#畫筆Pen繪制自定義線的帽子

    這篇文章主要介紹了C#畫筆Pen繪制自定義線的帽子,實例分析了畫筆Pen的使用技巧,需要的朋友可以參考下
    2015-06-06
  • C# 使用BitBlt進行窗口抓圖的示例

    C# 使用BitBlt進行窗口抓圖的示例

    這篇文章主要介紹了C# 使用BitBlt進行窗口抓圖的示例,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下
    2021-01-01
  • UGUI實現(xiàn)ScrollView無限滾動效果

    UGUI實現(xiàn)ScrollView無限滾動效果

    這篇文章主要為大家詳細介紹了UGUI實現(xiàn)ScrollView無限滾動效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-02-02
  • C#實現(xiàn)簡單訂單管理程序

    C#實現(xiàn)簡單訂單管理程序

    這篇文章主要為大家詳細介紹了C#實現(xiàn)簡單訂單管理程序,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • WPF使用Dragablz構(gòu)建可拖拽分離的Tab頁程序

    WPF使用Dragablz構(gòu)建可拖拽分離的Tab頁程序

    這篇文章介紹了WPF使用Dragablz構(gòu)建可拖拽分離Tab頁的方法,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • 使用WPF實現(xiàn)一個虛擬鍵盤的代碼示例

    使用WPF實現(xiàn)一個虛擬鍵盤的代碼示例

    在某些特定場景下,我們可能需要使用虛擬鍵盤來替代實體鍵盤,本文將詳細介紹如何使用 WPF 來實現(xiàn)一個虛擬鍵盤,并監(jiān)控鍵盤輸入,從而達到完全替代實體鍵盤的目的,需要的朋友可以參考下
    2025-04-04
  • 淺談C#設(shè)計模式之開放封閉原則

    淺談C#設(shè)計模式之開放封閉原則

    這篇文章主要介紹了淺談C#設(shè)計模式之開放封閉原則,需要的朋友可以參考下
    2014-12-12
  • C#播放鈴聲最簡單實現(xiàn)方法

    C#播放鈴聲最簡單實現(xiàn)方法

    這篇文章主要介紹了C#播放鈴聲最簡單實現(xiàn)方法,通過調(diào)用系統(tǒng)方法實現(xiàn)播放wav格式音頻文件的功能,是非常實用的技巧,需要的朋友可以參考下
    2014-12-12

最新評論