Redis并發(fā)問(wèn)題解決方案
前言
在多個(gè)客戶(hù)端并發(fā)訪(fǎng)問(wèn)Redis的時(shí)候,雖然Redis是單線(xiàn)程執(zhí)行指令,但是由于客戶(hù)端指令達(dá)到Redis的時(shí)序無(wú)法保證,所以可能出現(xiàn)如下的情況,導(dǎo)致并發(fā)問(wèn)題。
2個(gè)客戶(hù)端都執(zhí)行 get, set指令,期望將key的值設(shè)置為3,結(jié)果因?yàn)椴l(fā)問(wèn)題,導(dǎo)致結(jié)果為2 client1 get x => 1 client2 get x => 1 client1 set x => 2 client2 set x => 2
本文介紹 Redis 并發(fā)方面的解決方案。
Redis 的單個(gè)命令是原子的,但是一個(gè)業(yè)務(wù)操作可能包含多條命令,比如以下場(chǎng)景:客戶(hù)端查詢(xún)值,并遞增,在高并發(fā)場(chǎng)景下就可能出現(xiàn)并發(fā)問(wèn)題,導(dǎo)致數(shù)據(jù)不一致。
為了保證并發(fā)訪(fǎng)問(wèn)的正確性,Redis 提供了三種方法,原子操作、分布式鎖、事務(wù)。
1.分布式鎖
與分布式鎖相對(duì)的是本地鎖,假如只有一個(gè)服務(wù)實(shí)例,就可以直接在該單應(yīng)用本地使用鎖變量來(lái)控制多個(gè)客戶(hù)端的訪(fǎng)問(wèn)。
如果使用的是多實(shí)例的分布式系統(tǒng),就需要使用分布式鎖,即將鎖保存在一個(gè)第三方的共享存儲(chǔ)系統(tǒng)中,可以被多個(gè)客戶(hù)端共享訪(fǎng)問(wèn)和獲取。通常將一個(gè) Redis 實(shí)例作為分布式鎖的存儲(chǔ)系統(tǒng)。
實(shí)現(xiàn)分布式鎖的關(guān)鍵在于:
- 保證每個(gè)加鎖、釋放鎖操作都是原子的;
- 保證共享存儲(chǔ)系統(tǒng)的可靠性,即鎖的可靠性;
分布式鎖相較于 Lua 腳本,更簡(jiǎn)單易用,但是可能存在死鎖問(wèn)題。分布式鎖的性能不如 Lua 腳本。
1.基于單個(gè)節(jié)點(diǎn)
Redis 提供了 SETNX 命令(在 SET 命令后加上 NX 選項(xiàng)也能達(dá)到同樣的效果),保證了加鎖操作的原子性。
同時(shí),為了避免客戶(hù)端加鎖后不釋放,應(yīng)該給鎖變量設(shè)置過(guò)期時(shí)間(set NX EX),且在過(guò)期釋放鎖時(shí),判斷業(yè)務(wù)代碼是否執(zhí)行完成,如果未完成則給鎖續(xù)期。如果多次續(xù)期后,業(yè)務(wù)仍然未完成,再釋放鎖。(仍然存在風(fēng)險(xiǎn))
為了區(qū)分不同客戶(hù)端的操作,應(yīng)該將鎖變量設(shè)置為隨機(jī)值或唯一值,在釋放鎖時(shí)進(jìn)行驗(yàn)證。
釋放鎖的邏輯包含了讀取鎖變量、判斷值、刪除鎖變量的多個(gè)操作,所以應(yīng)該使用 Lua 腳本來(lái)保證互斥執(zhí)行。
單個(gè)節(jié)點(diǎn)可以實(shí)現(xiàn)分布式鎖的功能,但是無(wú)法保證可靠性。
2.基于多個(gè)節(jié)點(diǎn)
為了避免 Redis 實(shí)例故障而導(dǎo)致的鎖無(wú)法工作的問(wèn)題,Redis 的開(kāi)發(fā)者 Antirez 提出了分布式鎖算法 Redlock。
Redlock 算法的基本思路,是讓客戶(hù)端和多個(gè)獨(dú)立的 Redis 實(shí)例依次請(qǐng)求加鎖,如果客戶(hù)端能夠和半數(shù)以上的實(shí)例成功地完成加鎖操作,就認(rèn)為客戶(hù)端成功地獲得分布式鎖了,否則加鎖失敗。這樣一來(lái),即使有單個(gè)實(shí)例發(fā)生故障,因?yàn)殒i變量在其它實(shí)例上也有保存,所以客戶(hù)端仍然可以正常地進(jìn)行鎖操作,鎖變量并不會(huì)丟失。
加鎖過(guò)程:
- 客戶(hù)端獲取當(dāng)前時(shí)間;
- 客戶(hù)端按順序依次向 N 個(gè) Redis 實(shí)例執(zhí)行加鎖操作。同樣使用 SETNX 命令,并設(shè)置超時(shí)時(shí)間。如果請(qǐng)求加鎖一直超時(shí),則視為加鎖失敗,向下一個(gè)實(shí)例執(zhí)行加鎖操作。
- 客戶(hù)端完成所有加鎖操作后,計(jì)算整個(gè)加鎖過(guò)程的總耗時(shí)。
客戶(hù)端只有在滿(mǎn)足下面的這兩個(gè)條件時(shí),才能認(rèn)為是加鎖成功:
- 從超過(guò)半數(shù)(大于等于 N/2+1)的實(shí)例上成功獲取到了鎖;
- 獲取鎖的總耗時(shí)沒(méi)有超過(guò)鎖的有效時(shí)間。
在滿(mǎn)足了這兩個(gè)條件后,還需要重新計(jì)算這把鎖的有效時(shí)間,計(jì)算的結(jié)果是鎖的最初有效時(shí)間減去客戶(hù)端為獲取鎖的總耗時(shí)。如果鎖的有效時(shí)間已經(jīng)來(lái)不及完成共享數(shù)據(jù)的操作了,可以釋放鎖,以免出現(xiàn)還沒(méi)完成數(shù)據(jù)操作,鎖就過(guò)期了的情況。
如果沒(méi)能同時(shí)滿(mǎn)足這兩個(gè)條件,則視為加鎖失敗,執(zhí)行釋放鎖的過(guò)程:客戶(hù)端會(huì)向所有節(jié)點(diǎn)發(fā)起釋放鎖的操作,執(zhí)行釋放鎖的 Lua 腳本。
判斷是否加鎖時(shí),需要查詢(xún)所有節(jié)點(diǎn),以半數(shù)以上節(jié)點(diǎn)的鎖狀態(tài)來(lái)判斷整個(gè)分布式鎖的狀態(tài)。在釋放鎖之前,需要先判斷分布式鎖的狀態(tài)。
為了避免 Redis 節(jié)點(diǎn)發(fā)生崩潰重啟后造成鎖丟失,從而影響鎖的安全性,antirez 還提出了延時(shí)重啟的概念,即一個(gè)節(jié)點(diǎn)崩潰后不要立即重啟,而是等待一段時(shí)間后再進(jìn)行重啟,這段時(shí)間應(yīng)該大于鎖的有效時(shí)間。優(yōu)點(diǎn)是保證了鎖不會(huì)被多個(gè)客戶(hù)端獲?。蝗秉c(diǎn)是延長(zhǎng)了重啟時(shí)間,可能對(duì)系統(tǒng)造成影響。
性能和一致性是沖突的,如果為了分布式鎖的高可用性,可以開(kāi)啟持久化,但是會(huì)有額外的性能開(kāi)銷(xiāo),需要根據(jù)實(shí)際場(chǎng)景進(jìn)行選擇。
3.watch(樂(lè)觀(guān)鎖)
watch通常跟redis事務(wù)配合使用,watch某個(gè)key在操作過(guò)程中有沒(méi)有被其他指令改變,進(jìn)而做出相應(yīng)的處理。底層利用了CAS操作,后面講Redis事務(wù)會(huì)講到。
2.原子操作
為了實(shí)現(xiàn)并發(fā)控制要求的臨界區(qū)代碼互斥執(zhí)行,Redis 的原子操作采用了兩種方法:?jiǎn)蚊畈僮骱?Lua 腳本。
1.單命令操作
Redis 的每個(gè)操作都是原子性的。
Redis 是使用單線(xiàn)程來(lái)串行處理客戶(hù)端的請(qǐng)求操作命令的,所以,當(dāng) Redis 執(zhí)行某個(gè)命令操作時(shí),其他命令是無(wú)法執(zhí)行的,這相當(dāng)于單個(gè)操作是原子的。雖然 Redis 的單個(gè)操作是原子的,但是通常修改數(shù)據(jù)是包含多個(gè)操作的,至少包括讀數(shù)據(jù)、修改數(shù)據(jù)、寫(xiě)回?cái)?shù)據(jù)這三個(gè)操作,此時(shí)仍然可能出現(xiàn)并發(fā)問(wèn)題。
針對(duì)常用的修改數(shù)據(jù)場(chǎng)景,Redis 提供了 INCR/DECR 命令,可以對(duì)數(shù)據(jù)進(jìn)行簡(jiǎn)單的遞增/遞減操作,它們本身就是單個(gè)命令操作,在執(zhí)行時(shí),具有互斥性。但是如果要執(zhí)行更復(fù)雜的操作,Redis 的單命令操作就無(wú)法保證互斥執(zhí)行了。
2.Lua 腳本(多命令操作)
Redis 可以將多個(gè)操作寫(xiě)在 Lua 腳本中,然后把整個(gè) Lua 腳本作為一個(gè)整體執(zhí)行,在執(zhí)行的過(guò)程中不會(huì)被其他命令打斷,從而保證了 Lua 腳本中操作的原子性。
為什么是 Lua 腳本,而不是其他語(yǔ)言的腳本?
Lua 是一種高效的輕量級(jí) 腳本語(yǔ)言,用標(biāo)準(zhǔn) C 語(yǔ)言編寫(xiě)并以源代碼形式開(kāi)放。其設(shè)計(jì)目的就是為了嵌入應(yīng)用程序中,從而為應(yīng)用程序提供靈活的擴(kuò)展和定制功能。
Lua 腳本可以在服務(wù)器端執(zhí)行,不需要將數(shù)據(jù)傳輸?shù)娇蛻?hù)端再進(jìn)行處理,可以減少網(wǎng)絡(luò)傳輸?shù)拈_(kāi)銷(xiāo),因此性能較高。
使用 Lua 腳本不僅可以實(shí)現(xiàn)將多個(gè)操作原子執(zhí)行,還能夠復(fù)用 Lua 腳本。但是使用 Lua 腳本需要額外的語(yǔ)言學(xué)習(xí)成本,還有調(diào)試?yán)щy、可讀性較差的問(wèn)題。
如果把很多操作都放在 Lua 腳本中原子執(zhí)行,會(huì)導(dǎo)致 Redis 執(zhí)行腳本的時(shí)間增加,同樣也會(huì)降低 Redis 的并發(fā)性能。所以,在編寫(xiě) Lua 腳本時(shí),要避免把不需要做并發(fā)控制的操作寫(xiě)入腳本中。
在 Lua 腳本執(zhí)行過(guò)程中崩潰怎么辦?
Redis 會(huì)在內(nèi)部維護(hù)一個(gè)已經(jīng)加載腳本的 哈希表,記錄了每個(gè)腳本的 SHA1 值和對(duì)應(yīng)的 Lua 腳本代碼。當(dāng) Redis 服務(wù)器重啟時(shí),Redis 會(huì)自動(dòng)重新加載這個(gè)哈希表中記錄的所有腳本,再重新執(zhí)行,此時(shí)可能導(dǎo)致部分修改被應(yīng)用。所以 Lua 腳本并不能?chē)?yán)格保證原子性。如果對(duì)數(shù)據(jù)一致性非常嚴(yán)格,可以使用 Lua 腳本+事務(wù) WATCH 的辦法。
3.事務(wù)
Redis 事務(wù)的本質(zhì)是一組命令的集合。事務(wù)支持一次執(zhí)行多個(gè)命令,一個(gè)事務(wù)中所有命令都會(huì)被序列化。在事務(wù)執(zhí)行過(guò)程,會(huì)按照順序串行化執(zhí)行隊(duì)列中的命令,其他客戶(hù)端提交的命令請(qǐng)求不會(huì)插入到事務(wù)執(zhí)行命令序列中。
Redis 提供了實(shí)現(xiàn)事務(wù)的幾個(gè)命令:
- MULTI :開(kāi)啟事務(wù),redis 會(huì)將后續(xù)的命令逐個(gè)放入隊(duì)列中,然后使用 EXEC 命令來(lái)原子化執(zhí)行這個(gè)命令系列。
- EXEC:執(zhí)行事務(wù)中的所有操作命令。
- DISCARD:取消事務(wù),放棄執(zhí)行事務(wù)塊中的所有命令。
- WATCH:在開(kāi)啟事務(wù)之前監(jiān)視一個(gè)或多個(gè) key,如果事務(wù)在執(zhí)行前,這個(gè) key (或多個(gè) key)被其他命令修改,則事務(wù)被中斷,不會(huì)執(zhí)行事務(wù)中的任何命令(一般需要在 EXEC 執(zhí)行失敗后重新執(zhí)行整個(gè)函數(shù))。
UNWATCH:取消 WATCH 對(duì)所有 key 的監(jiān)視。
為什么 WATCH 是中斷事務(wù),而不是阻塞其他進(jìn)程?這樣不會(huì)導(dǎo)致并發(fā)量高的時(shí)候,被 WATCH 的事務(wù)一直得不到執(zhí)行嗎?
這種機(jī)制稱(chēng)為 樂(lè)觀(guān)鎖,因?yàn)樵诖蠖鄶?shù)情況下,碰撞的概率很小,所以選用了更容易實(shí)現(xiàn)的方式(且影響不大)。

在使用事務(wù)時(shí),可以配合 Pipeline 使用:一次性將所有命令打包好,再全部發(fā)送到服務(wù)端。
相比于事務(wù)的入隊(duì),同樣是一次性執(zhí)行,這樣不僅能減少網(wǎng)絡(luò) IO,還能保證在開(kāi)啟 WATCH 時(shí)不會(huì)被其他操作打斷。
1.執(zhí)行步驟
- 開(kāi)啟事務(wù):使用 MULTI 命令開(kāi)啟事務(wù);
- 入隊(duì):接收到命令后并不會(huì)立即執(zhí)行,而是放到等待執(zhí)行的事務(wù)隊(duì)列里;
- 執(zhí)行:由 EXEC 命令觸發(fā)事務(wù)執(zhí)行。
當(dāng)客戶(hù)端切換到事務(wù)狀態(tài)之后, 服務(wù)器會(huì)根據(jù)這個(gè)客戶(hù)端發(fā)來(lái)的不同命令執(zhí)行不同的操作:
- 如果客戶(hù)端發(fā)送的命令為 EXEC 、DISCARD、WATCH、MULTI 四個(gè)命令的其中一個(gè), 那么服務(wù)器立即執(zhí)行這個(gè)命令;
- 如果是其他命令, 那么服務(wù)器并不立即執(zhí)行命令, 而是將這個(gè)命令放入一個(gè)事務(wù)隊(duì)列里面, 然后向客戶(hù)端返回 QUEUED 回復(fù);
2.錯(cuò)誤處理
在事務(wù)執(zhí)行過(guò)程中可能遇到兩種不同類(lèi)型的錯(cuò)誤,會(huì)有不同的應(yīng)對(duì)方案:
- 編譯器錯(cuò)誤:命令在編譯時(shí)出錯(cuò),會(huì)導(dǎo)致整個(gè)事務(wù)提交失敗,即所有命令執(zhí)行不成功;
- 運(yùn)行時(shí)錯(cuò)誤:命令在運(yùn)行時(shí)檢測(cè)到錯(cuò)誤,最終會(huì)導(dǎo)致事務(wù)提交失敗,但是事務(wù)并不會(huì)回滾,而是跳過(guò)錯(cuò)誤命令繼續(xù)執(zhí)行并保留結(jié)果;
為什么 Redis 不支持 事務(wù)回滾?
Redis 命令只會(huì)因?yàn)殄e(cuò)誤的語(yǔ)法而失敗(并且這些問(wèn)題不能在入隊(duì)時(shí)發(fā)現(xiàn)),或是命令用在了錯(cuò)誤類(lèi)型的鍵上面:這也就是說(shuō),從實(shí)用性的角度來(lái)說(shuō),失敗的命令是由編程錯(cuò)誤造成的,而這些錯(cuò)誤應(yīng)該在開(kāi)發(fā)的過(guò)程中被發(fā)現(xiàn),而不應(yīng)該出現(xiàn)在生產(chǎn)環(huán)境中。
不需要對(duì)回滾進(jìn)行支持,所以 Redis 的內(nèi)部可以保持簡(jiǎn)單且快速。
3.崩潰處理
Redis 在執(zhí)行事務(wù)時(shí)會(huì)使用一個(gè)單獨(dú)的內(nèi)存空間來(lái)保存事務(wù)中的所有修改操作,只有當(dāng)事務(wù)成功提交時(shí),這些修改操作才會(huì)被應(yīng)用到 Redis 中。因此,如果事務(wù)被中止,所有的修改操作也都會(huì)被撤銷(xiāo),從而保證了數(shù)據(jù)的一致性。
如果開(kāi)啟了 AOF 持久化,會(huì)先將事務(wù)中的所有命令寫(xiě)入 AOF 緩沖區(qū),然后執(zhí)行事務(wù)中的命令,再將 AOF 緩沖區(qū)中的數(shù)據(jù)寫(xiě)入到 AOF 文件。
如果在寫(xiě)入 AOF 文件前崩潰,則持久化失敗,相當(dāng)于事務(wù)沒(méi)有發(fā)生,不會(huì)出現(xiàn)數(shù)據(jù)不一致。
另外,RDB 快照不會(huì)在事務(wù)執(zhí)行途中進(jìn)行。
總結(jié)
本文介紹了 Redis 應(yīng)對(duì)并發(fā)問(wèn)題的三種方案,Redis 中的單條命令都是原子操作,而且還有 INCR/DECR 來(lái)應(yīng)對(duì)簡(jiǎn)單的場(chǎng)景。對(duì)于復(fù)雜的場(chǎng)景,Redis 可以使用 Lua 腳本、分布式鎖、事務(wù)來(lái)實(shí)現(xiàn)操作的原子性。Lua 腳本是將一系列操作放在一個(gè)腳本中原子執(zhí)行。分布式鎖是通過(guò)共享的鎖變量來(lái)限制客戶(hù)端的并發(fā)訪(fǎng)問(wèn)。事務(wù)是將一系列操作放到執(zhí)行隊(duì)列中,再按順序原子執(zhí)行。
到此這篇關(guān)于Redis并發(fā)問(wèn)題解決方案的文章就介紹到這了,更多相關(guān)Redis并發(fā)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 記錄一次并發(fā)情況下的redis導(dǎo)致服務(wù)假死的問(wèn)題解決
- 高并發(fā)場(chǎng)景分析之redis+lua防重校驗(yàn)
- Redis處理高并發(fā)之布隆過(guò)濾器詳解
- Redis并發(fā)訪(fǎng)問(wèn)問(wèn)題詳細(xì)講解
- 淺談Redis如何應(yīng)對(duì)并發(fā)訪(fǎng)問(wèn)
- Redis高并發(fā)情況下并發(fā)扣減庫(kù)存項(xiàng)目實(shí)戰(zhàn)
- Redis高并發(fā)場(chǎng)景下秒殺超賣(mài)解決方案(秒殺場(chǎng)景)
- redis 解決庫(kù)存并發(fā)問(wèn)題實(shí)現(xiàn)數(shù)量控制
- 高并發(fā)下Redis如何保持?jǐn)?shù)據(jù)一致性(避免讀后寫(xiě))
相關(guān)文章
如何保證Redis與數(shù)據(jù)庫(kù)的數(shù)據(jù)一致性
這篇文章主要介紹了如何保證Redis與數(shù)據(jù)庫(kù)的數(shù)據(jù)一致性,文中舉了兩個(gè)場(chǎng)景例子介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05
Redis序列化轉(zhuǎn)換類(lèi)型報(bào)錯(cuò)的解決
本文主要介紹了Redis序列化轉(zhuǎn)換類(lèi)型報(bào)錯(cuò)的解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04
Redis發(fā)布訂閱和實(shí)現(xiàn).NET客戶(hù)端詳解
發(fā)布訂閱在應(yīng)用級(jí)其作用是為了減少依賴(lài)關(guān)系,通常也叫觀(guān)察者模式。主要是把耦合點(diǎn)單獨(dú)抽離出來(lái)作為第三方,隔離易變化的發(fā)送方和接收方。下面這篇文章主要給大家介紹了關(guān)于Redis發(fā)布訂閱和實(shí)現(xiàn).NET客戶(hù)端的相關(guān)資料,需要的朋友可以參考下2017-03-03
Redis設(shè)置密碼的實(shí)現(xiàn)步驟
本文主要介紹了Redis設(shè)置密碼的實(shí)現(xiàn)步驟,主要包括兩種方法:臨時(shí)密碼和持久密碼,具有一定的參考價(jià)值,感興趣的可以了解一下2023-08-08
Window下對(duì)Redis進(jìn)行開(kāi)啟與關(guān)閉的操作方法
這篇文章主要介紹了Window下對(duì)Redis進(jìn)行開(kāi)啟與關(guān)閉的操作方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-11-11

