成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Redis鎖被別人釋放怎么辦

存儲(chǔ) 存儲(chǔ)軟件 Redis
當(dāng)多個(gè)進(jìn)程不在同一個(gè)系統(tǒng)中,用分布式鎖控制多個(gè)進(jìn)程對資源的訪問。有這樣一個(gè)情境,線程A和線程B都共享某個(gè)變量X。如果是分布式情況下,線程A和線程B很可能不是在同一對象中,每個(gè)客戶端在釋放鎖時(shí),都是刪除操作,并沒有檢查這把鎖是否還是自己的,所以就會(huì)發(fā)生釋放別人鎖的風(fēng)險(xiǎn)。

 [[427025]]

本文轉(zhuǎn)載自微信公眾號(hào)「后端Q」,作者conan 。轉(zhuǎn)載本文請聯(lián)系后端Q公眾號(hào)。

什么是分布式鎖?

要介紹分布式鎖,首先要提到與分布式鎖相對應(yīng)的是線程鎖、進(jìn)程鎖。

線程鎖:主要用來給方法、代碼塊加鎖。當(dāng)某個(gè)方法或代碼使用鎖,在同一時(shí)刻僅有一個(gè)線程執(zhí)行該方法或該代碼段。線程鎖只在同一JVM中有效果,因?yàn)榫€程鎖的實(shí)現(xiàn)在根本上是依靠線程之間共享內(nèi)存實(shí)現(xiàn)的,比如synchronized是共享對象頭,顯示鎖Lock是共享某個(gè)變量(state)。

進(jìn)程鎖:為了控制同一操作系統(tǒng)中多個(gè)進(jìn)程訪問某個(gè)共享資源,因?yàn)檫M(jìn)程具有獨(dú)立性,各個(gè)進(jìn)程無法訪問其他進(jìn)程的資源,因此無法通過synchronized等線程鎖實(shí)現(xiàn)進(jìn)程鎖。

問題窺探

分布式鎖:當(dāng)多個(gè)進(jìn)程不在同一個(gè)系統(tǒng)中,用分布式鎖控制多個(gè)進(jìn)程對資源的訪問。有這樣一個(gè)情境,線程A和線程B都共享某個(gè)變量X。如果是分布式情況下,線程A和線程B很可能不是在同一對象中,每個(gè)客戶端在釋放鎖時(shí),都是刪除操作,并沒有檢查這把鎖是否還是自己的,所以就會(huì)發(fā)生釋放別人鎖的風(fēng)險(xiǎn)。

解決辦法

客戶端在加鎖時(shí),設(shè)置一個(gè)只有自己知道的唯一標(biāo)識(shí)進(jìn)去。例如,可以是自己的線程 ID,也可以是一個(gè) UUID(隨機(jī)且唯一),這里我們以 UUID 舉例:

  1. // 鎖的VALUE設(shè)置為UUID 
  2. 127.0.0.1:6379> SET lock $uuid EX 20 NX 
  3. OK 

這里假設(shè) 20s 操作共享時(shí)間完全足夠,先不考慮鎖自動(dòng)過期的問題。之后,在釋放鎖時(shí),要先判斷這把鎖是否還歸自己持有,偽代碼可以這么寫:

  1. // 鎖是自己的,才釋放 
  2. if redis.get("lock") == $uuid: 
  3.     redis.del("lock"
  4.      

這里釋放鎖使用的是 GET + DEL 兩條命令,這時(shí),又會(huì)遇到我們前面講的原子性問題了。

客戶端 1 執(zhí)行 GET,判斷鎖是自己的

客戶端 2 執(zhí)行了 SET 命令,強(qiáng)制獲取到鎖(雖然發(fā)生概率比較低,但我們需要嚴(yán)謹(jǐn)?shù)乜紤]鎖的安全性模型)

客戶端 1 執(zhí)行 DEL,卻釋放了客戶端 2 的鎖

由此可見,這兩個(gè)命令還是必須要原子執(zhí)行才行。

怎樣原子執(zhí)行呢?Lua 腳本。

我們可以把這個(gè)邏輯,寫成 Lua 腳本,讓 Redis 來執(zhí)行。

因?yàn)?Redis 處理每一個(gè)請求是單線程執(zhí)行的,在執(zhí)行一個(gè) Lua 腳本時(shí),其它請求必須等待,直到這個(gè) Lua 腳本處理完成,這樣一來,GET + DEL 之間就不會(huì)插入其它命令了。安全釋放鎖的 Lua 腳本如下:

  1. // 判斷鎖是自己的,才釋放 
  2. if redis.call("GET",KEYS[1]) == ARGV[1] 
  3. then 
  4.     return redis.call("DEL",KEYS[1]) 
  5. else 
  6.     return 0 
  7. end 

好了,這樣一路優(yōu)化,整個(gè)的加鎖、解鎖的流程就更嚴(yán)謹(jǐn)了。

這里我們先小結(jié)一下,基于 Redis 實(shí)現(xiàn)的分布式鎖,一個(gè)嚴(yán)謹(jǐn)?shù)牡牧鞒倘缦拢?/p>

  1. 加鎖:SET lock_key $unique_id EX $expire_time NX 

操作共享資源 釋放鎖:Lua 腳本,先 GET 判斷鎖是否歸屬自己,再 DEL 釋放鎖

責(zé)任編輯:武曉燕 來源: 后端Q
相關(guān)推薦

2024-03-13 13:25:09

Redis分布式鎖

2021-04-13 10:41:25

Redis內(nèi)存數(shù)據(jù)庫

2019-10-12 09:50:46

Redis內(nèi)存數(shù)據(jù)庫

2021-08-07 05:05:30

接口Redis項(xiàng)目

2011-06-30 17:58:30

網(wǎng)站被K

2020-07-10 08:46:26

HTTPS證書劫持網(wǎng)絡(luò)協(xié)議

2024-08-06 08:08:14

2024-10-18 09:55:50

RedisHash數(shù)據(jù)

2024-10-09 17:06:52

RedisHash哈希表

2012-11-27 10:41:33

2015-03-31 15:33:55

2022-08-24 08:17:14

RedisRDBAOF

2021-01-05 10:48:38

RedisAOF日志RDB快照

2017-05-11 16:54:16

2011-06-27 15:42:23

降權(quán)SEO

2015-10-28 17:09:13

技術(shù)創(chuàng)業(yè)

2018-01-30 09:25:04

2022-09-05 09:02:01

服務(wù)器CPU服務(wù)

2017-03-13 15:25:51

Windows 7Windows端口占用

2021-01-26 08:02:04

Redis內(nèi)存數(shù)據(jù)庫
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 美女视频一区二区三区 | 中文字幕一级 | 亚洲一区电影 | 午夜视频在线免费观看 | 成人黄色av网站 | 免费大黄视频 | 成人免费淫片aa视频免费 | 亚洲精品一区在线观看 | 日本在线观看网址 | 欧美极品一区二区 | 久久精品亚洲成在人线av网址 | 色婷婷综合久久久中字幕精品久久 | 国产乱码精品1区2区3区 | 久久免费国产 | 国产人成精品一区二区三 | 国产有码 | 涩色视频在线观看 | 欧美日韩国产精品 | 亚洲91视频 | 国产亚洲高清视频 | 福利精品| 亚洲一区二区三区免费观看 | 自拍在线 | 精精国产xxxx视频在线野外 | 国产精品我不卡 | 午夜在线视频 | 精品九九 | 日韩视频中文字幕 | 国产日韩免费观看 | 日本小电影在线 | 日韩视频精品在线 | 国产一区不卡 | 找个黄色片 | 成人教育av | 亚洲在线一区 | 亚洲精品乱码久久久久久久久久 | 日韩欧美一区二区三区免费看 | 日韩三级在线观看 | 久久精品国产久精国产 | 国产精品一区二区视频 | 亚洲精品一区二区网址 |