国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

百度社招面試題——如何用Redis實現(xiàn)分布式鎖

YanceyOfficial / 3542人閱讀

摘要:集群實現(xiàn)分布式鎖上面的討論中我們有一個非常重要的假設(shè)是單點的。但是其實這已經(jīng)超出了實現(xiàn)分布式鎖的范圍,單純用沒有命令來實現(xiàn)生成。這個問題用實現(xiàn)分布式鎖暫時無解。結(jié)論并不能實現(xiàn)嚴格意義上的分布式鎖。

關(guān)于Redis實現(xiàn)分布式鎖的問題,網(wǎng)絡(luò)上很多,但是很多人的討論基本就是把原來博主的貼過來,甚至很多面試官也是一知半解經(jīng)不起推敲就來面候選人,最近結(jié)合我自己的學(xué)習(xí)和資料查閱,整理一下用Redis實現(xiàn)分布式鎖的方法,歡迎評論、交流、討論。

1.單機Redis實現(xiàn)分布式鎖 1.1獲取鎖

獲取鎖的過程很簡單,客戶端向Redis發(fā)送命令:

SET resource_name my_random_value NX PX 30000

my_random_value是由客戶端生成的一個隨機字符串,它要保證在足夠長的一段時間內(nèi)在所有客戶端的所有獲取鎖的請求中都是唯一的。
NX表示只有當(dāng)resource_name對應(yīng)的key值不存在的時候才能SET成功。這保證了只有第一個請求的客戶端才能獲得鎖,而其它客戶端在鎖被釋放之前都無法獲得鎖。
PX 30000表示這個鎖有一個30秒的自動過期時間。

1.2 釋放鎖
if redis.call("get",KEYS[1]) == ARGV[1] then
    return redis.call("del",KEYS[1])
else
    return 0
end

之前獲取鎖的時候生成的my_random_value 作為參數(shù)傳到Lua腳本里面,作為:ARGV[1],而 resource_name作為KEYS[1]。Lua腳本可以保證操作的原子性。

1.3 關(guān)于單點Redis實現(xiàn)分布式鎖的討論

網(wǎng)絡(luò)上有文章說用如下命令獲取鎖:

SETNX resource_name my_random_value
EXPIRE resource_name 30

由于這兩個命令不是原子的。如果客戶端在執(zhí)行完SETNXcrash了,那么就沒有機會執(zhí)行EXPIRE了,導(dǎo)致它一直持有這個鎖,其他的客戶端就永遠獲取不到這個鎖了。

為什么my_random_value 要設(shè)置成隨機值?

保證了一個客戶端釋放的鎖是自己持有的那個鎖。如若不然,可能出現(xiàn)鎖不安全的情況。

客戶端1獲取鎖成功。
客戶端1在某個操作上阻塞了很長時間。
過期時間到了,鎖自動釋放了。
客戶端2獲取到了對應(yīng)同一個資源的鎖。
客戶端1從阻塞中恢復(fù)過來,釋放掉了客戶端2持有的鎖。

用 SETNX獲取鎖

網(wǎng)上大量文章說用如下命令獲取鎖:

SETNX lock.foo 

原文在Redis對SETNX的官網(wǎng)說明,Redis官網(wǎng)文檔建議用Set命令來代替,主要原因是SETNX不支持超時時間的設(shè)置。

https://redis.io/commands/setnx

2.Redis集群實現(xiàn)分布式鎖

上面的討論中我們有一個非常重要的假設(shè):Redis是單點的。如果Redis是集群模式,我們考慮如下場景:

客戶端1從Master獲取了鎖。
Master宕機了,存儲鎖的key還沒有來得及同步到Slave上。
Slave升級為Master。
客戶端2從新的Master獲取到了對應(yīng)同一個資源的鎖。
客戶端1和客戶端2同時持有了同一個資源的鎖,鎖不再具有安全性。

就此問題,Redis作者antirez寫了RedLock算法來解決這種問題。

2.1 RedLock獲取鎖

獲取當(dāng)前時間。

按順序依次向N個Redis節(jié)點執(zhí)行獲取鎖的操作。這個獲取操作跟前面基于單Redis節(jié)點的獲取鎖的過程相同,包含隨機字符串my_random_value,也包含過期時間(比如PX 30000,即鎖的有效時間)。為了保證在某個Redis節(jié)點不可用的時候算法能夠繼續(xù)運行,這個獲取鎖的操作還有一個超時時間(time out),它要遠小于鎖的有效時間(幾十毫秒量級)。客戶端在向某個Redis節(jié)點獲取鎖失敗以后,應(yīng)該立即嘗試下一個Redis節(jié)點。

計算整個獲取鎖的過程總共消耗了多長時間,計算方法是用當(dāng)前時間減去第1步記錄的時間。如果客戶端從大多數(shù)Redis節(jié)點(>= N/2+1)成功獲取到了鎖,并且獲取鎖總共消耗的時間沒有超過鎖的有效時間(lock validity time),那么這時客戶端才認為最終獲取鎖成功;否則,認為最終獲取鎖失敗。

如果最終獲取鎖成功了,那么這個鎖的有效時間應(yīng)該重新計算,它等于最初的鎖的有效時間減去第3步計算出來的獲取鎖消耗的時間。

如果最終獲取鎖失敗了(可能由于獲取到鎖的Redis節(jié)點個數(shù)少于N/2+1,或者整個獲取鎖的過程消耗的時間超過了鎖的最初有效時間),那么客戶端應(yīng)該立即向所有Redis節(jié)點發(fā)起釋放鎖的操作(即前面介紹的單機Redis Lua腳本釋放鎖的方法)。

2.2 RedLock釋放鎖

客戶端向所有Redis節(jié)點發(fā)起釋放鎖的操作,不管這些節(jié)點當(dāng)時在獲取鎖的時候成功與否。

2.3 關(guān)于RedLock的問題討論

如果有節(jié)點發(fā)生崩潰重啟

假設(shè)一共有5個Redis節(jié)點:A, B, C, D, E。設(shè)想發(fā)生了如下的事件序列:

客戶端1成功鎖住了A, B, C,獲取鎖成功(但D和E沒有鎖住)。
節(jié)點C崩潰重啟了,但客戶端1在C上加的鎖沒有持久化下來,丟失了。
節(jié)點C重啟后,客戶端2鎖住了C, D, E,獲取鎖成功。
客戶端1和客戶端2同時獲得了鎖。

為了應(yīng)對這一問題,antirez又提出了延遲重啟(delayed restarts)的概念。也就是說,一個節(jié)點崩潰后,先不立即重啟它,而是等待一段時間再重啟,這段時間應(yīng)該大于鎖的有效時間(lock validity time)。這樣的話,這個節(jié)點在重啟前所參與的鎖都會過期,它在重啟后就不會對現(xiàn)有的鎖造成影響。

如果客戶端長期阻塞導(dǎo)致鎖過期

解釋一下這個時序圖,客戶端1在獲得鎖之后發(fā)生了很長時間的GC pause,在此期間,它獲得的鎖過期了,而客戶端2獲得了鎖。當(dāng)客戶端1從GC pause中恢復(fù)過來的時候,它不知道自己持有的鎖已經(jīng)過期了,它依然向共享資源(上圖中是一個存儲服務(wù))發(fā)起了寫數(shù)據(jù)請求,而這時鎖實際上被客戶端2持有,因此兩個客戶端的寫請求就有可能沖突(鎖的互斥作用失效了)。

如何解決這個問題呢?引入了fencing token的概念:

客戶端1先獲取到的鎖,因此有一個較小的fencing token,等于33,而客戶端2后獲取到的鎖,有一個較大的fencing token,等于34。客戶端1從GC pause中恢復(fù)過來之后,依然是向存儲服務(wù)發(fā)送訪問請求,但是帶了fencing token = 33。存儲服務(wù)發(fā)現(xiàn)它之前已經(jīng)處理過34的請求,所以會拒絕掉這次33的請求。這樣就避免了沖突。

但是其實這已經(jīng)超出了Redis實現(xiàn)分布式鎖的范圍,單純用Redis沒有命令來實現(xiàn)生成Token。

時鐘跳躍問題

假設(shè)有5個Redis節(jié)點A, B, C, D, E。

客戶端1從Redis節(jié)點A, B, C成功獲取了鎖(多數(shù)節(jié)點)。由于網(wǎng)絡(luò)問題,與D和E通信失敗。
節(jié)點C上的時鐘發(fā)生了向前跳躍,導(dǎo)致它上面維護的鎖快速過期。
客戶端2從Redis節(jié)點C, D, E成功獲取了同一個資源的鎖(多數(shù)節(jié)點)。
客戶端1和客戶端2現(xiàn)在都認為自己持有了鎖。
這個問題用Redis實現(xiàn)分布式鎖暫時無解。而生產(chǎn)環(huán)境這種情況是存在的。

結(jié)論
Redis并不能實現(xiàn)嚴格意義上的分布式鎖。但是這并不意味著上面討論的方案一無是處。如果你的應(yīng)用場景為了效率(efficiency),協(xié)調(diào)各個客戶端避免做重復(fù)的工作,即使鎖失效了,只是可能把某些操作多做一遍而已,不會產(chǎn)生其它的不良后果。但是如果你的應(yīng)用場景是為了正確性(correctness),那么用Redis實現(xiàn)分布式鎖并不合適,會存在各種各樣的問題,且解決起來就很復(fù)雜,為了正確性,需要使用zab、raft共識算法,或者使用帶有事務(wù)的數(shù)據(jù)庫來實現(xiàn)嚴格意義上的分布式鎖。

參考資料
Distributed locks with Redis
基于Redis的分布式鎖到底安全嗎(上)? - 鐵蕾的個人博客
https://martin.kleppmann.com/...

熱門閱讀

技術(shù)文章匯總

【Leetcode】101. 對稱二叉樹

【Leetcode】100. 相同的樹

【Leetcode】98. 驗證二叉搜索樹

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/11998.html

相關(guān)文章

  • 三年百度,五年阿里,阿里p8架構(gòu)師淺談:我是何順利進入BAT

    摘要:三年百度,五年阿里,阿里架構(gòu)師淺談我是如何順利進入前些天在我群里認識了以為挺有意思的老哥,他也是工作年多技術(shù)和面試都不差,最近也是在找工作,是從京城來魔都的,也和他撈了不少。 說來慚愧,也不怕你們笑話。做開發(fā)8年多,到目前還是一名不折不扣的掃地僧。年前的辭職,到現(xiàn)在還在家靜養(yǎng)中。其實也沒什么,就是回家總結(jié)一下自己這些年來在外工作與面試等做一個簡單的總結(jié)與反思。做一下自己后面一個人生規(guī)劃...

    seanlook 評論0 收藏0
  • 【推薦】最新200篇:技術(shù)文章整理

    摘要:作為面試官,我是如何甄別應(yīng)聘者的包裝程度語言和等其他語言的對比分析和主從復(fù)制的原理詳解和持久化的原理是什么面試中經(jīng)常被問到的持久化與恢復(fù)實現(xiàn)故障恢復(fù)自動化詳解哨兵技術(shù)查漏補缺最易錯過的技術(shù)要點大掃盲意外宕機不難解決,但你真的懂?dāng)?shù)據(jù)恢復(fù)嗎每秒 作為面試官,我是如何甄別應(yīng)聘者的包裝程度Go語言和Java、python等其他語言的對比分析 Redis和MySQL Redis:主從復(fù)制的原理詳...

    BicycleWarrior 評論0 收藏0

發(fā)表評論

0條評論

YanceyOfficial

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<