为什么多个线程不可能同时抢到一把锁_HFL Redis_12_redis分布式锁的3种实现方式...
HotFrameLearning(簡稱 HFL) Redis_12_redis分布式鎖的3種實(shí)現(xiàn)方式
-
一、大致介紹
```
1、昨天介紹完redis的數(shù)據(jù)結(jié)構(gòu)后,有小伙伴讓本人講講redis的分布式鎖,因此才有了此篇的由來,只是把我的節(jié)奏提前了而已;
2、接下來我就通過幾種方式介紹一下分布式鎖的使用;
```
二、源碼分享
2.1 獲取redis鎖且快速失敗
圖1 - 獲取redis鎖且快速失敗
```
1、圖1中,我們首先區(qū)分的是單機(jī)模式還是集群模式;
2、通過簡單的 set 操作,但是要注意一點(diǎn),set 時需要設(shè)置 expiryTimeMills 超時時間,而且設(shè)置key與設(shè)置expiryTimeMills需要是在一條原子語句里面,如果不是的話則會引起不必要的死鎖問題;
3、至于為什么會引起死鎖,假設(shè) set key 與 set expiryTimeMills 是兩條代碼語句,當(dāng) set key 成功后,程序立馬宕機(jī)了,那么這個 key 就沒有失效時間,相當(dāng)于永久有效,那么下次再次競爭獲取該 key 時就永遠(yuǎn)都是失敗的,因此要特別注意key與expiryTimeMills需要在同一條原子語句里面被執(zhí)行;
4、當(dāng)時又有人說,嘗試一次就失敗了,豈不是太不讓調(diào)用方省心了,調(diào)用方想嘗試多次還得自己寫個代碼玩玩,因此就有了下面嘗試多次獲取鎖的操作;
```
2.2 嘗試數(shù)次獲取鎖,獲取不到則返回失敗
圖2 - 嘗試數(shù)次獲取鎖,獲取不到則返回失敗
```
1、圖2中,同樣我們首先區(qū)分的是單機(jī)模式還是集群模式;
2、通過在 tryLockFailFast 外面再套一層 while 循環(huán)處理,讓入?yún)⒍鄠魅雵L試次數(shù)retryTimes、嘗試間隔時間retryIntervalMills 兩個字段,即可滿足多次嘗試獲取鎖的訴求,這樣調(diào)用方就能省心了;
3、雖然加鎖是完事了,但是解鎖呢,是不是直接刪掉就完事了呢?請接著看~
```
2.3 釋放redis鎖且快速失敗
圖3 - 釋放redis鎖且快速失敗
```
1、圖3中,同樣我們首先區(qū)分的是單機(jī)模式還是集群模式;
2、只是在解鎖的時候,有點(diǎn)點(diǎn)不同,我們需要先看看我們解鎖的key對應(yīng)的內(nèi)存值是不是我們當(dāng)初加鎖的那個入?yún)⒅?#xff1f;
3、為什么要這么判斷呢?原因就是假設(shè)A線程加鎖了,但是A的任務(wù)還沒有執(zhí)行完,此時key超時過期了,然后B線程又成功搶到了該鎖,但是當(dāng)B前腳搶到鎖后A后腳就執(zhí)行完了任務(wù)準(zhǔn)備釋放鎖,那么就會將B已經(jīng)獲取到的鎖釋放掉,這么一操作已不是把不該釋放的釋放了,最終都會導(dǎo)致相應(yīng)業(yè)務(wù)功能執(zhí)行出問題的;
4、因此才會在解鎖的時候,看下鎖對應(yīng)的值是不是當(dāng)初加鎖的那個值,如果一致則刪除,否則刪除失敗;
5、但是這么一連串的 get del 是兩個命令語句,但是解鎖的真實(shí)訴求需要保證原子性,因此我們就需要 redis 給我們提供的 lua 腳本原子性執(zhí)行多個命令語句;
6、但是但是,說到這,又會有小伙伴會問,怕?lián)娜蝿?wù)執(zhí)行太久導(dǎo)致redis超時,有啥好的方式處理呢?接著看~
```
2.4 阻塞獲取redisson鎖
圖4 - 阻塞獲取redisson鎖
```
1、圖4中,用的redisson框架版本號為 redisson-3.11.2.jar,具體maven坐標(biāo)大家去中央倉庫看吧;
2、大家會想,這個redisson和之前的 tryUnLockFailFast 的加鎖有啥區(qū)別呢?不也是執(zhí)行加鎖么?
3、答案當(dāng)然不一樣,redisson 底層會有一個續(xù)命線程,每到過期時間的的1/3時就自動重新將過期時間置為入?yún)⒌倪^期時間值,所以即使業(yè)務(wù)線程的任務(wù)執(zhí)行很久,也不會因?yàn)殒i自動過期一事而煩惱,豈不快哉;
```
2.5 redisson知識擴(kuò)展
圖5 - redisson知識擴(kuò)展
```
1、圖5中,我們稍微多看一眼 redisson 框架,然后我們會發(fā)現(xiàn)底層提供了很多的鎖操作;
2、然而稍微再認(rèn)真的同學(xué)仔細(xì)看看,會發(fā)現(xiàn)我們 JUC 有的類,這里都差不多實(shí)現(xiàn)了一番,可見 redisson 為分布式做了很多封裝,方便調(diào)用方盡情的使用;
```
歡迎關(guān)注+點(diǎn)贊,您的肯定是對我最大的支持!!!
總結(jié)
以上是生活随笔為你收集整理的为什么多个线程不可能同时抢到一把锁_HFL Redis_12_redis分布式锁的3种实现方式...的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 你发这些什么目的_微信CRM系统究竟是什
- 下一篇: pymysql.err.Internal